-
Notifications
You must be signed in to change notification settings - Fork 6
/
Copy pathec2-alarms.ts
114 lines (106 loc) · 5.32 KB
/
ec2-alarms.ts
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
import { Duration } from "aws-cdk-lib";
import { ComparisonOperator, MathExpression, Statistic, TreatMissingData } from "aws-cdk-lib/aws-cloudwatch";
import { HttpCodeElb, HttpCodeTarget } from "aws-cdk-lib/aws-elasticloadbalancingv2";
import type { GuStack } from "../core";
import { AppIdentity } from "../core";
import type { GuApplicationLoadBalancer, GuApplicationTargetGroup } from "../loadbalancing";
import { GuAlarm } from "./alarm";
import type { GuAlarmProps, Http4xxAlarmProps, Http5xxAlarmProps } from "./alarm";
interface GuAlb4xxPercentageAlarmProps extends Pick<GuAlarmProps, "snsTopicName">, Http4xxAlarmProps, AppIdentity {
loadBalancer: GuApplicationLoadBalancer;
}
interface GuAlb5xxPercentageAlarmProps extends Pick<GuAlarmProps, "snsTopicName">, Http5xxAlarmProps, AppIdentity {
loadBalancer: GuApplicationLoadBalancer;
}
interface GuUnhealthyInstancesAlarmProps extends Pick<GuAlarmProps, "snsTopicName" | "actionsEnabled">, AppIdentity {
targetGroup: GuApplicationTargetGroup;
}
/**
* Creates an alarm which is triggered whenever the percentage of requests with a 5xx response code exceeds
* the specified threshold.
*/
export class GuAlb5xxPercentageAlarm extends GuAlarm {
constructor(scope: GuStack, props: GuAlb5xxPercentageAlarmProps) {
const mathExpression = new MathExpression({
expression: "100*(m1+m2)/m3",
usingMetrics: {
m1: props.loadBalancer.metrics.httpCodeElb(HttpCodeElb.ELB_5XX_COUNT),
m2: props.loadBalancer.metrics.httpCodeTarget(HttpCodeTarget.TARGET_5XX_COUNT),
m3: props.loadBalancer.metrics.requestCount(),
},
label: `% of 5XX responses served for ${props.app} (load balancer and instances combined)`,
period: Duration.minutes(1),
});
const defaultAlarmName = `High 5XX error percentage from ${props.app} in ${scope.stage}`;
const defaultDescription = `${props.app} exceeded ${props.tolerated5xxPercentage}% error rate`;
const alarmProps = {
...props,
metric: mathExpression,
treatMissingData: TreatMissingData.NOT_BREACHING,
threshold: props.tolerated5xxPercentage,
comparisonOperator: ComparisonOperator.GREATER_THAN_THRESHOLD,
alarmName: props.alarmName ?? defaultAlarmName,
alarmDescription: props.alarmDescription ?? defaultDescription,
evaluationPeriods: props.numberOfMinutesAboveThresholdBeforeAlarm ?? 1,
};
super(scope, AppIdentity.suffixText(props, "High5xxPercentageAlarm"), alarmProps);
}
}
/**
* Creates an alarm which is triggered whenever the percentage of requests with a 4xx response code exceeds
* the specified threshold.
*/
export class GuAlb4xxPercentageAlarm extends GuAlarm {
constructor(scope: GuStack, props: GuAlb4xxPercentageAlarmProps) {
const mathExpression = new MathExpression({
expression: "100*(m1+m2)/m3",
usingMetrics: {
m1: props.loadBalancer.metrics.httpCodeElb(HttpCodeElb.ELB_4XX_COUNT),
m2: props.loadBalancer.metrics.httpCodeTarget(HttpCodeTarget.TARGET_4XX_COUNT),
m3: props.loadBalancer.metrics.requestCount(),
},
label: `% of 4XX responses served for ${props.app} (load balancer and instances combined)`,
period: Duration.minutes(1),
});
const defaultAlarmName = `High 4XX error percentage from ${props.app} in ${scope.stage}`;
const defaultDescription = `${props.app} exceeded ${props.tolerated4xxPercentage}% error rate`;
const alarmProps = {
...props,
metric: mathExpression,
treatMissingData: TreatMissingData.NOT_BREACHING,
threshold: props.tolerated4xxPercentage,
comparisonOperator: ComparisonOperator.GREATER_THAN_THRESHOLD,
alarmName: props.alarmName ?? defaultAlarmName,
alarmDescription: props.alarmDescription ?? defaultDescription,
evaluationPeriods: props.numberOfMinutesAboveThresholdBeforeAlarm ?? 1,
};
super(scope, AppIdentity.suffixText(props, "High4xxPercentageAlarm"), alarmProps);
}
}
/**
* Creates an alarm which is triggered whenever there have been several healthcheck failures within a single hour.
*/
export class GuUnhealthyInstancesAlarm extends GuAlarm {
constructor(scope: GuStack, props: GuUnhealthyInstancesAlarmProps) {
const alarmName = `Unhealthy instances for ${props.app} in ${scope.stage}`;
const period = Duration.minutes(1);
const evaluationPeriods = 60;
const evaluationInterval = Duration.minutes(period.toMinutes() * evaluationPeriods).toHumanString();
const alarmDescription = `${props.app}'s instances have failed healthchecks several times over the last ${evaluationInterval}.
This typically results in the AutoScaling Group cycling instances and can lead to problems with deployment,
scaling or handling traffic spikes.
Check ${props.app}'s application logs or ssh onto an unhealthy instance in order to debug these problems.`;
const alarmProps: GuAlarmProps = {
...props,
alarmName: alarmName,
alarmDescription: alarmDescription,
metric: props.targetGroup.metrics.unhealthyHostCount().with({ period, statistic: Statistic.MAXIMUM }),
treatMissingData: TreatMissingData.NOT_BREACHING,
threshold: 1,
comparisonOperator: ComparisonOperator.GREATER_THAN_OR_EQUAL_TO_THRESHOLD,
datapointsToAlarm: 30,
evaluationPeriods,
};
super(scope, AppIdentity.suffixText(props, "UnhealthyInstancesAlarm"), alarmProps);
}
}