@liflig/cdk 3.22.19 → 3.23.1
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/assets/README.md +32 -0
- package/assets/slack-error-log-handler-lambda/.python-version +1 -0
- package/assets/slack-error-log-handler-lambda/__pycache__/index.cpython-314.pyc +0 -0
- package/assets/slack-error-log-handler-lambda/__pycache__/test_handler.cpython-314-pytest-9.0.2.pyc +0 -0
- package/assets/slack-error-log-handler-lambda/index.py +411 -0
- package/assets/slack-error-log-handler-lambda/pyproject.toml +13 -0
- package/assets/slack-error-log-handler-lambda/test_handler.py +229 -0
- package/assets/slack-error-log-handler-lambda/uv.lock +176 -0
- package/lib/alarms/database-alarms.d.ts +31 -18
- package/lib/alarms/database-alarms.js +39 -18
- package/lib/alarms/index.d.ts +2 -0
- package/lib/alarms/index.js +2 -1
- package/lib/alarms/lambda-alarms.d.ts +116 -14
- package/lib/alarms/lambda-alarms.js +158 -29
- package/lib/alarms/log-filter-patterns.d.ts +5 -0
- package/lib/alarms/log-filter-patterns.js +6 -0
- package/lib/alarms/queue-alarms.d.ts +88 -0
- package/lib/alarms/queue-alarms.js +109 -0
- package/lib/alarms/service-alarms.d.ts +75 -8
- package/lib/alarms/service-alarms.js +130 -38
- package/lib/alarms/slack-alarm.d.ts +6 -1
- package/lib/alarms/slack-alarm.js +23 -5
- package/lib/ecs/fargate-service.d.ts +111 -1
- package/lib/ecs/fargate-service.js +46 -2
- package/lib/load-balancer/load-balancer.js +2 -2
- package/lib/rds/database.d.ts +83 -0
- package/lib/rds/database.js +41 -1
- package/package.json +1 -1
|
@@ -1,41 +1,143 @@
|
|
|
1
1
|
import * as cloudwatch from "aws-cdk-lib/aws-cloudwatch";
|
|
2
2
|
import type * as lambda from "aws-cdk-lib/aws-lambda";
|
|
3
|
+
import * as logs from "aws-cdk-lib/aws-logs";
|
|
3
4
|
import * as constructs from "constructs";
|
|
4
5
|
export interface LambdaAlarmsProps {
|
|
5
6
|
/**
|
|
6
|
-
* The
|
|
7
|
+
* The CloudWatch alarm action to use for high-severity alarms (ALARM channel).
|
|
7
8
|
*/
|
|
8
|
-
|
|
9
|
+
alarmAction: cloudwatch.IAlarmAction;
|
|
10
|
+
/**
|
|
11
|
+
* The CloudWatch alarm action to use for lower-severity warnings (WARNING channel).
|
|
12
|
+
*/
|
|
13
|
+
warningAction: cloudwatch.IAlarmAction;
|
|
9
14
|
/**
|
|
10
15
|
* The Lambda to add alarms to.
|
|
11
16
|
*/
|
|
12
17
|
lambdaFunction: lambda.IFunction;
|
|
18
|
+
/**
|
|
19
|
+
* Optional Lambda function that will receive forwarded log events.
|
|
20
|
+
* If provided, subscription filters will be created to forward matching logs
|
|
21
|
+
*/
|
|
22
|
+
logHandler?: lambda.IFunction;
|
|
13
23
|
}
|
|
14
24
|
/**
|
|
15
25
|
* Alarms for Lambda functions.
|
|
16
26
|
*
|
|
17
|
-
*
|
|
18
|
-
*
|
|
27
|
+
* Unlike RDS and ECS alarm constructs in this package, `LambdaAlarms` is
|
|
28
|
+
* set up manually by consumers.
|
|
19
29
|
*
|
|
20
|
-
*
|
|
21
|
-
*
|
|
30
|
+
* Defaults:
|
|
31
|
+
* - single invocation (singleError) -> sent to `warningAction` by default
|
|
32
|
+
* - multiple invocation (multipleErrors) -> sent to `alarmAction` by default
|
|
33
|
+
* - error log alarm (ERROR/FATAL JSON logs) -> sent to `warningAction` by default
|
|
34
|
+
* - uncaught Java exception alarm -> sent to `warningAction` by default (disabled by default)
|
|
22
35
|
*/
|
|
23
36
|
export declare class LambdaAlarms extends constructs.Construct {
|
|
24
|
-
private readonly
|
|
37
|
+
private readonly alarmAction;
|
|
38
|
+
private readonly warningAction;
|
|
25
39
|
private readonly lambdaFunction;
|
|
40
|
+
private readonly logHandler?;
|
|
26
41
|
constructor(scope: constructs.Construct, id: string, props: LambdaAlarmsProps);
|
|
27
42
|
/**
|
|
28
|
-
*
|
|
29
|
-
*
|
|
43
|
+
* Creates both the single-invocation warning and the
|
|
44
|
+
* multiple invocations alarm with reasonable defaults.
|
|
45
|
+
*
|
|
46
|
+
* Each created alarm can be overridden via the per-alarm props.
|
|
30
47
|
*/
|
|
31
|
-
addInvocationErrorAlarm(
|
|
48
|
+
addInvocationErrorAlarm(props?: {
|
|
49
|
+
singleError?: {
|
|
50
|
+
appendToAlarmDescription?: string;
|
|
51
|
+
enabled?: boolean;
|
|
52
|
+
action?: cloudwatch.IAlarmAction;
|
|
53
|
+
/**
|
|
54
|
+
* @default true
|
|
55
|
+
*/
|
|
56
|
+
enableOkAlarm?: boolean;
|
|
57
|
+
};
|
|
58
|
+
multipleErrors?: {
|
|
59
|
+
appendToAlarmDescription?: string;
|
|
60
|
+
enabled?: boolean;
|
|
61
|
+
action?: cloudwatch.IAlarmAction;
|
|
62
|
+
/**
|
|
63
|
+
* @default true
|
|
64
|
+
*/
|
|
65
|
+
enableOkAlarm?: boolean;
|
|
66
|
+
/**
|
|
67
|
+
* @default 3
|
|
68
|
+
*/
|
|
69
|
+
evaluationPeriods?: number;
|
|
70
|
+
/**
|
|
71
|
+
* @default 1
|
|
72
|
+
*/
|
|
73
|
+
threshold?: number;
|
|
74
|
+
/**
|
|
75
|
+
* @default 60
|
|
76
|
+
*/
|
|
77
|
+
periodSeconds?: number;
|
|
78
|
+
};
|
|
79
|
+
}): void;
|
|
32
80
|
/**
|
|
33
|
-
*
|
|
81
|
+
* Sets up a CloudWatch Alarm that triggers on a single invocation failure.
|
|
34
82
|
*/
|
|
35
|
-
props?: {
|
|
83
|
+
addSingleInvocationAlarm(props?: {
|
|
84
|
+
appendToAlarmDescription?: string;
|
|
85
|
+
enabled?: boolean;
|
|
86
|
+
action?: cloudwatch.IAlarmAction;
|
|
87
|
+
enableOkAlarm?: boolean;
|
|
88
|
+
}): void;
|
|
89
|
+
/**
|
|
90
|
+
* Sets up a CloudWatch Alarm that triggers when there is a short series of
|
|
91
|
+
* invocation failures.
|
|
92
|
+
*/
|
|
93
|
+
addMultipleInvocationAlarms(props?: {
|
|
94
|
+
appendToAlarmDescription?: string;
|
|
95
|
+
enabled?: boolean;
|
|
96
|
+
action?: cloudwatch.IAlarmAction;
|
|
36
97
|
/**
|
|
37
|
-
*
|
|
98
|
+
* @default true
|
|
38
99
|
*/
|
|
39
|
-
|
|
100
|
+
enableOkAlarm?: boolean;
|
|
101
|
+
/**
|
|
102
|
+
* @default 3
|
|
103
|
+
*/
|
|
104
|
+
evaluationPeriods?: number;
|
|
105
|
+
/**
|
|
106
|
+
* @default 1
|
|
107
|
+
*/
|
|
108
|
+
threshold?: number;
|
|
109
|
+
/**
|
|
110
|
+
* @default 60
|
|
111
|
+
*/
|
|
112
|
+
periodSeconds?: number;
|
|
113
|
+
}): void;
|
|
114
|
+
/**
|
|
115
|
+
* Alerts when ERROR is logged from lambda.
|
|
116
|
+
*/
|
|
117
|
+
addErrorAlarm(props: {
|
|
118
|
+
logGroup: logs.ILogGroup;
|
|
119
|
+
alarmDescription?: string;
|
|
120
|
+
/**
|
|
121
|
+
* @default true
|
|
122
|
+
*/
|
|
123
|
+
enableOkAlarm?: boolean;
|
|
124
|
+
/**
|
|
125
|
+
* An action to use for CloudWatch alarm state changes
|
|
126
|
+
* instead of the default warning action
|
|
127
|
+
*/
|
|
128
|
+
action?: cloudwatch.IAlarmAction;
|
|
129
|
+
}): void;
|
|
130
|
+
/**
|
|
131
|
+
* Adds an alarm for uncaught Java exceptions in the Lambda function's logs.
|
|
132
|
+
*/
|
|
133
|
+
addUncaughtJavaExceptionAlarm(props: {
|
|
134
|
+
logGroup: logs.ILogGroup;
|
|
135
|
+
alarmDescription?: string;
|
|
136
|
+
/**
|
|
137
|
+
* @default false
|
|
138
|
+
*/
|
|
139
|
+
enabled?: boolean;
|
|
140
|
+
enableOkAlarm?: boolean;
|
|
141
|
+
action?: cloudwatch.IAlarmAction;
|
|
40
142
|
}): void;
|
|
41
143
|
}
|
|
@@ -1,49 +1,178 @@
|
|
|
1
1
|
import * as cdk from "aws-cdk-lib";
|
|
2
2
|
import * as cloudwatch from "aws-cdk-lib/aws-cloudwatch";
|
|
3
|
+
import * as logs from "aws-cdk-lib/aws-logs";
|
|
4
|
+
import * as logsDestinations from "aws-cdk-lib/aws-logs-destinations";
|
|
3
5
|
import * as constructs from "constructs";
|
|
6
|
+
import { jsonErrorFilterPattern } from "./log-filter-patterns";
|
|
4
7
|
/**
|
|
5
8
|
* Alarms for Lambda functions.
|
|
6
9
|
*
|
|
7
|
-
*
|
|
8
|
-
*
|
|
10
|
+
* Unlike RDS and ECS alarm constructs in this package, `LambdaAlarms` is
|
|
11
|
+
* set up manually by consumers.
|
|
9
12
|
*
|
|
10
|
-
*
|
|
11
|
-
*
|
|
13
|
+
* Defaults:
|
|
14
|
+
* - single invocation (singleError) -> sent to `warningAction` by default
|
|
15
|
+
* - multiple invocation (multipleErrors) -> sent to `alarmAction` by default
|
|
16
|
+
* - error log alarm (ERROR/FATAL JSON logs) -> sent to `warningAction` by default
|
|
17
|
+
* - uncaught Java exception alarm -> sent to `warningAction` by default (disabled by default)
|
|
12
18
|
*/
|
|
13
19
|
export class LambdaAlarms extends constructs.Construct {
|
|
14
|
-
|
|
20
|
+
alarmAction;
|
|
21
|
+
warningAction;
|
|
15
22
|
lambdaFunction;
|
|
23
|
+
logHandler;
|
|
16
24
|
constructor(scope, id, props) {
|
|
17
25
|
super(scope, id);
|
|
18
|
-
this.
|
|
26
|
+
this.alarmAction = props.alarmAction;
|
|
27
|
+
this.warningAction = props.warningAction;
|
|
19
28
|
this.lambdaFunction = props.lambdaFunction;
|
|
29
|
+
this.logHandler = props.logHandler;
|
|
20
30
|
}
|
|
21
31
|
/**
|
|
22
|
-
*
|
|
23
|
-
*
|
|
32
|
+
* Creates both the single-invocation warning and the
|
|
33
|
+
* multiple invocations alarm with reasonable defaults.
|
|
34
|
+
*
|
|
35
|
+
* Each created alarm can be overridden via the per-alarm props.
|
|
24
36
|
*/
|
|
25
|
-
addInvocationErrorAlarm(
|
|
37
|
+
addInvocationErrorAlarm(props) {
|
|
38
|
+
this.addSingleInvocationAlarm(props?.singleError);
|
|
39
|
+
this.addMultipleInvocationAlarms(props?.multipleErrors);
|
|
40
|
+
}
|
|
41
|
+
/**
|
|
42
|
+
* Sets up a CloudWatch Alarm that triggers on a single invocation failure.
|
|
43
|
+
*/
|
|
44
|
+
addSingleInvocationAlarm(props) {
|
|
45
|
+
if (props?.enabled !== false) {
|
|
46
|
+
// Sent to warnings channel by default
|
|
47
|
+
const action = props?.action ?? this.warningAction;
|
|
48
|
+
const alarm = new cloudwatch.Metric({
|
|
49
|
+
metricName: "Errors",
|
|
50
|
+
namespace: "AWS/Lambda",
|
|
51
|
+
statistic: "Sum",
|
|
52
|
+
period: cdk.Duration.seconds(60),
|
|
53
|
+
dimensionsMap: {
|
|
54
|
+
FunctionName: this.lambdaFunction.functionName,
|
|
55
|
+
},
|
|
56
|
+
}).createAlarm(this, "SingleInvocationAlarm", {
|
|
57
|
+
alarmDescription: `Invocation for '${this.lambdaFunction.functionName}' had a single failure. ${props?.appendToAlarmDescription ?? ""}`,
|
|
58
|
+
comparisonOperator: cloudwatch.ComparisonOperator.GREATER_THAN_OR_EQUAL_TO_THRESHOLD,
|
|
59
|
+
evaluationPeriods: 1,
|
|
60
|
+
threshold: 1,
|
|
61
|
+
treatMissingData: cloudwatch.TreatMissingData.IGNORE,
|
|
62
|
+
});
|
|
63
|
+
alarm.addAlarmAction(action);
|
|
64
|
+
if (props?.enableOkAlarm ?? true)
|
|
65
|
+
alarm.addOkAction(action);
|
|
66
|
+
}
|
|
67
|
+
}
|
|
68
|
+
/**
|
|
69
|
+
* Sets up a CloudWatch Alarm that triggers when there is a short series of
|
|
70
|
+
* invocation failures.
|
|
71
|
+
*/
|
|
72
|
+
addMultipleInvocationAlarms(props) {
|
|
73
|
+
if (props?.enabled !== false) {
|
|
74
|
+
const alarm = new cloudwatch.Metric({
|
|
75
|
+
metricName: "Errors",
|
|
76
|
+
namespace: "AWS/Lambda",
|
|
77
|
+
statistic: "Sum",
|
|
78
|
+
period: cdk.Duration.seconds(props?.periodSeconds ?? 60),
|
|
79
|
+
dimensionsMap: {
|
|
80
|
+
FunctionName: this.lambdaFunction.functionName,
|
|
81
|
+
},
|
|
82
|
+
}).createAlarm(this, "InvocationSeriesAlarm", {
|
|
83
|
+
alarmDescription: `Invocation for '${this.lambdaFunction.functionName}' failed repeatedly. ${props?.appendToAlarmDescription ?? ""}`,
|
|
84
|
+
comparisonOperator: cloudwatch.ComparisonOperator.GREATER_THAN_OR_EQUAL_TO_THRESHOLD,
|
|
85
|
+
evaluationPeriods: props?.evaluationPeriods ?? 3,
|
|
86
|
+
threshold: props?.threshold ?? 1,
|
|
87
|
+
treatMissingData: cloudwatch.TreatMissingData.IGNORE,
|
|
88
|
+
});
|
|
89
|
+
// Sent to alarm channel by default
|
|
90
|
+
const multipleAction = props?.action ?? this.alarmAction;
|
|
91
|
+
alarm.addAlarmAction(multipleAction);
|
|
92
|
+
if (props?.enableOkAlarm ?? true) {
|
|
93
|
+
alarm.addOkAction(multipleAction);
|
|
94
|
+
}
|
|
95
|
+
}
|
|
96
|
+
}
|
|
97
|
+
/**
|
|
98
|
+
* Alerts when ERROR is logged from lambda.
|
|
99
|
+
*/
|
|
100
|
+
addErrorAlarm(props) {
|
|
101
|
+
// If no log handler is configured, create simple "ERROR" metric alarm
|
|
102
|
+
if (!this.logHandler) {
|
|
103
|
+
const errorMetricFilter = props.logGroup.addMetricFilter("ErrorMetricFilter", {
|
|
104
|
+
filterPattern: jsonErrorFilterPattern(),
|
|
105
|
+
metricName: "Errors",
|
|
106
|
+
metricNamespace: `stack/${cdk.Stack.of(this).stackName}/${this.lambdaFunction.functionName}/Errors`,
|
|
107
|
+
});
|
|
108
|
+
const errorAlarm = errorMetricFilter
|
|
109
|
+
.metric()
|
|
110
|
+
.with({
|
|
111
|
+
statistic: "Sum",
|
|
112
|
+
period: cdk.Duration.seconds(60),
|
|
113
|
+
})
|
|
114
|
+
.createAlarm(this, "ErrorLogAlarm", {
|
|
115
|
+
alarmDescription: props.alarmDescription ??
|
|
116
|
+
`${this.lambdaFunction.functionName} logged an error`,
|
|
117
|
+
evaluationPeriods: 1,
|
|
118
|
+
threshold: 1,
|
|
119
|
+
treatMissingData: cloudwatch.TreatMissingData.NOT_BREACHING,
|
|
120
|
+
});
|
|
121
|
+
// Sent to warnings channel by default
|
|
122
|
+
const action = props.action ?? this.warningAction;
|
|
123
|
+
errorAlarm.addAlarmAction(action);
|
|
124
|
+
if (props.enableOkAlarm ?? true) {
|
|
125
|
+
errorAlarm.addOkAction(action);
|
|
126
|
+
}
|
|
127
|
+
}
|
|
128
|
+
if (this.logHandler) {
|
|
129
|
+
props.logGroup.addSubscriptionFilter("liflig-cdk-log-content-to-slack-error-subscription", {
|
|
130
|
+
destination: new logsDestinations.LambdaDestination(this.logHandler),
|
|
131
|
+
filterPattern: jsonErrorFilterPattern(),
|
|
132
|
+
});
|
|
133
|
+
}
|
|
134
|
+
}
|
|
26
135
|
/**
|
|
27
|
-
*
|
|
136
|
+
* Adds an alarm for uncaught Java exceptions in the Lambda function's logs.
|
|
28
137
|
*/
|
|
29
|
-
props) {
|
|
30
|
-
|
|
31
|
-
|
|
32
|
-
|
|
33
|
-
|
|
34
|
-
|
|
35
|
-
|
|
36
|
-
|
|
37
|
-
|
|
38
|
-
|
|
39
|
-
|
|
40
|
-
|
|
41
|
-
|
|
42
|
-
|
|
43
|
-
|
|
44
|
-
|
|
45
|
-
|
|
46
|
-
|
|
138
|
+
addUncaughtJavaExceptionAlarm(props) {
|
|
139
|
+
if (props.enabled) {
|
|
140
|
+
const filterPattern = logs.FilterPattern.allTerms("Exception in thread");
|
|
141
|
+
// If no log handler is configured, create a simple metric alarm.
|
|
142
|
+
if (!this.logHandler) {
|
|
143
|
+
const errorMetricFilter = props.logGroup.addMetricFilter("UncaughtJavaExceptionFilter", {
|
|
144
|
+
filterPattern: filterPattern,
|
|
145
|
+
metricName: "UncaughtJavaException",
|
|
146
|
+
metricNamespace: `stack/${cdk.Stack.of(this).stackName}/${this.lambdaFunction.functionName}/UncaughtJavaException`,
|
|
147
|
+
});
|
|
148
|
+
const errorAlarm = errorMetricFilter
|
|
149
|
+
.metric()
|
|
150
|
+
.with({
|
|
151
|
+
statistic: "Sum",
|
|
152
|
+
period: cdk.Duration.seconds(60),
|
|
153
|
+
})
|
|
154
|
+
.createAlarm(this, "UncaughtJavaExceptionLogAlarm", {
|
|
155
|
+
alarmDescription: props.alarmDescription ??
|
|
156
|
+
`${this.lambdaFunction.functionName} logged an uncaught Java exception`,
|
|
157
|
+
evaluationPeriods: 1,
|
|
158
|
+
threshold: 1,
|
|
159
|
+
treatMissingData: cloudwatch.TreatMissingData.NOT_BREACHING,
|
|
160
|
+
});
|
|
161
|
+
// Default to the warning action
|
|
162
|
+
const actionToUse = props.action ?? this.warningAction;
|
|
163
|
+
errorAlarm.addAlarmAction(actionToUse);
|
|
164
|
+
if (props.enableOkAlarm ?? true) {
|
|
165
|
+
errorAlarm.addOkAction(actionToUse);
|
|
166
|
+
}
|
|
167
|
+
}
|
|
168
|
+
// If a log handler is configured, forward matching logs to it.
|
|
169
|
+
if (this.logHandler) {
|
|
170
|
+
props.logGroup.addSubscriptionFilter("liflig-cdk-log-content-to-slack-uncaught-exception-subscription", {
|
|
171
|
+
destination: new logsDestinations.LambdaDestination(this.logHandler),
|
|
172
|
+
filterPattern: filterPattern,
|
|
173
|
+
});
|
|
174
|
+
}
|
|
175
|
+
}
|
|
47
176
|
}
|
|
48
177
|
}
|
|
49
|
-
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoibGFtYmRhLWFsYXJtcy5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uLy4uL3NyYy9hbGFybXMvbGFtYmRhLWFsYXJtcy50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFBQSxPQUFPLEtBQUssR0FBRyxNQUFNLGFBQWEsQ0FBQTtBQUNsQyxPQUFPLEtBQUssVUFBVSxNQUFNLDRCQUE0QixDQUFBO0FBRXhELE9BQU8sS0FBSyxVQUFVLE1BQU0sWUFBWSxDQUFBO0FBYXhDOzs7Ozs7OztHQVFHO0FBQ0gsTUFBTSxPQUFPLFlBQWEsU0FBUSxVQUFVLENBQUMsU0FBUztJQUNuQyxNQUFNLENBQXlCO0lBQy9CLGNBQWMsQ0FBa0I7SUFFakQsWUFDRSxLQUEyQixFQUMzQixFQUFVLEVBQ1YsS0FBd0I7UUFFeEIsS0FBSyxDQUFDLEtBQUssRUFBRSxFQUFFLENBQUMsQ0FBQTtRQUVoQixJQUFJLENBQUMsTUFBTSxHQUFHLEtBQUssQ0FBQyxNQUFNLENBQUE7UUFDMUIsSUFBSSxDQUFDLGNBQWMsR0FBRyxLQUFLLENBQUMsY0FBYyxDQUFBO0lBQzVDLENBQUM7SUFFRDs7O09BR0c7SUFDSCx1QkFBdUI7SUFDckI7O09BRUc7SUFDSCxLQUtDO1FBRUQsTUFBTSxLQUFLLEdBQUcsSUFBSSxVQUFVLENBQUMsTUFBTSxDQUFDO1lBQ2xDLFVBQVUsRUFBRSxRQUFRO1lBQ3BCLFNBQVMsRUFBRSxZQUFZO1lBQ3ZCLFNBQVMsRUFBRSxLQUFLO1lBQ2hCLE1BQU0sRUFBRSxHQUFHLENBQUMsUUFBUSxDQUFDLE9BQU8sQ0FBQyxFQUFFLENBQUM7WUFDaEMsYUFBYSxFQUFFO2dCQUNiLFlBQVksRUFBRSxJQUFJLENBQUMsY0FBYyxDQUFDLFlBQVk7YUFDL0M7U0FDRixDQUFDLENBQUMsV0FBVyxDQUFDLElBQUksRUFBRSx1QkFBdUIsRUFBRTtZQUM1QyxnQkFBZ0IsRUFBRSxtQkFBbUIsSUFBSSxDQUFDLGNBQWMsQ0FBQyxZQUFZLGFBQWEsS0FBSyxFQUFFLHdCQUF3QixJQUFJLEVBQUUsRUFBRTtZQUN6SCxrQkFBa0IsRUFDaEIsVUFBVSxDQUFDLGtCQUFrQixDQUFDLGtDQUFrQztZQUNsRSxpQkFBaUIsRUFBRSxDQUFDO1lBQ3BCLFNBQVMsRUFBRSxDQUFDO1lBQ1osZ0JBQWdCLEVBQUUsVUFBVSxDQUFDLGdCQUFnQixDQUFDLE1BQU07U0FDckQsQ0FBQyxDQUFBO1FBRUYsS0FBSyxDQUFDLGNBQWMsQ0FBQyxJQUFJLENBQUMsTUFBTSxDQUFDLENBQUE7UUFDakMsS0FBSyxDQUFDLFdBQVcsQ0FBQyxJQUFJLENBQUMsTUFBTSxDQUFDLENBQUE7SUFDaEMsQ0FBQztDQUNGIiwic291cmNlc0NvbnRlbnQiOlsiaW1wb3J0ICogYXMgY2RrIGZyb20gXCJhd3MtY2RrLWxpYlwiXG5pbXBvcnQgKiBhcyBjbG91ZHdhdGNoIGZyb20gXCJhd3MtY2RrLWxpYi9hd3MtY2xvdWR3YXRjaFwiXG5pbXBvcnQgdHlwZSAqIGFzIGxhbWJkYSBmcm9tIFwiYXdzLWNkay1saWIvYXdzLWxhbWJkYVwiXG5pbXBvcnQgKiBhcyBjb25zdHJ1Y3RzIGZyb20gXCJjb25zdHJ1Y3RzXCJcblxuZXhwb3J0IGludGVyZmFjZSBMYW1iZGFBbGFybXNQcm9wcyB7XG4gIC8qKlxuICAgKiBUaGUgZGVmYXVsdCBhY3Rpb24gdG8gdXNlIGZvciBDbG91ZFdhdGNoIGFsYXJtIHN0YXRlIGNoYW5nZXMuXG4gICAqL1xuICBhY3Rpb246IGNsb3Vkd2F0Y2guSUFsYXJtQWN0aW9uXG4gIC8qKlxuICAgKiBUaGUgTGFtYmRhIHRvIGFkZCBhbGFybXMgdG8uXG4gICAqL1xuICBsYW1iZGFGdW5jdGlvbjogbGFtYmRhLklGdW5jdGlvblxufVxuXG4vKipcbiAqIEFsYXJtcyBmb3IgTGFtYmRhIGZ1bmN0aW9ucy5cbiAqXG4gKiBCeSBpdHNlbGYsIG5vIGFsYXJtcyBhcmUgY3JlYXRlZC4gVXNlIHRoZSBtZXRob2RzIGF2YWlsYWJsZVxuICogdG8gYWRkIGFsYXJtcy5cbiAqXG4gKiBDcmVhdGUgbXVsdGlwbGUgaW5zdGFuY2VzIG9mIHtAbGluayBMYW1iZGFBbGFybXN9IHdpdGggZGlmZmVyZW50IGBhY3Rpb25gXG4gKiBpZiB5b3UgbmVlZCBhbiBhbGFybSB0byBkbyBtdWx0aXBsZSB0aGluZ3MuXG4gKi9cbmV4cG9ydCBjbGFzcyBMYW1iZGFBbGFybXMgZXh0ZW5kcyBjb25zdHJ1Y3RzLkNvbnN0cnVjdCB7XG4gIHByaXZhdGUgcmVhZG9ubHkgYWN0aW9uOiBjbG91ZHdhdGNoLklBbGFybUFjdGlvblxuICBwcml2YXRlIHJlYWRvbmx5IGxhbWJkYUZ1bmN0aW9uOiBsYW1iZGEuSUZ1bmN0aW9uXG5cbiAgY29uc3RydWN0b3IoXG4gICAgc2NvcGU6IGNvbnN0cnVjdHMuQ29uc3RydWN0LFxuICAgIGlkOiBzdHJpbmcsXG4gICAgcHJvcHM6IExhbWJkYUFsYXJtc1Byb3BzLFxuICApIHtcbiAgICBzdXBlcihzY29wZSwgaWQpXG5cbiAgICB0aGlzLmFjdGlvbiA9IHByb3BzLmFjdGlvblxuICAgIHRoaXMubGFtYmRhRnVuY3Rpb24gPSBwcm9wcy5sYW1iZGFGdW5jdGlvblxuICB9XG5cbiAgLyoqXG4gICAqIFNldHMgdXAgYSBDbG91ZFdhdGNoIEFsYXJtIHRoYXQgdHJpZ2dlcnMgaWYgdGhlIExhbWJkYSBmYWlscyBpbnZvY2F0aW9ucy5cbiAgICogVGhpcyB1c3VhbGx5IGhhcHBlbnMgZnJvbSB1bmNhdWdodCBleGNlcHRpb25zIGluIHRoZSBsYW1iZGEuXG4gICAqL1xuICBhZGRJbnZvY2F0aW9uRXJyb3JBbGFybShcbiAgICAvKipcbiAgICAgKiBDb25maWd1cmF0aW9uIGZvciBhbiBhbGFybS5cbiAgICAgKi9cbiAgICBwcm9wcz86IHtcbiAgICAgIC8qKlxuICAgICAgICogQWRkIGV4dHJhIGluZm9ybWF0aW9uIHRvIHRoZSBhbGFybSBkZXNjcmlwdGlvbiwgbGlrZSBSdW5ib29rIFVSTCBvciBzdGVwcyB0byB0cmlhZ2UuXG4gICAgICAgKi9cbiAgICAgIGFwcGVuZFRvQWxhcm1EZXNjcmlwdGlvbj86IHN0cmluZ1xuICAgIH0sXG4gICk6IHZvaWQge1xuICAgIGNvbnN0IGFsYXJtID0gbmV3IGNsb3Vkd2F0Y2guTWV0cmljKHtcbiAgICAgIG1ldHJpY05hbWU6IFwiRXJyb3JzXCIsXG4gICAgICBuYW1lc3BhY2U6IFwiQVdTL0xhbWJkYVwiLFxuICAgICAgc3RhdGlzdGljOiBcIlN1bVwiLFxuICAgICAgcGVyaW9kOiBjZGsuRHVyYXRpb24uc2Vjb25kcyg2MCksXG4gICAgICBkaW1lbnNpb25zTWFwOiB7XG4gICAgICAgIEZ1bmN0aW9uTmFtZTogdGhpcy5sYW1iZGFGdW5jdGlvbi5mdW5jdGlvbk5hbWUsXG4gICAgICB9LFxuICAgIH0pLmNyZWF0ZUFsYXJtKHRoaXMsIFwiRmFpbGVkSW52b2NhdGlvbkFsYXJtXCIsIHtcbiAgICAgIGFsYXJtRGVzY3JpcHRpb246IGBJbnZvY2F0aW9uIGZvciAnJHt0aGlzLmxhbWJkYUZ1bmN0aW9uLmZ1bmN0aW9uTmFtZX0nIGZhaWxlZC4gJHtwcm9wcz8uYXBwZW5kVG9BbGFybURlc2NyaXB0aW9uID8/IFwiXCJ9YCxcbiAgICAgIGNvbXBhcmlzb25PcGVyYXRvcjpcbiAgICAgICAgY2xvdWR3YXRjaC5Db21wYXJpc29uT3BlcmF0b3IuR1JFQVRFUl9USEFOX09SX0VRVUFMX1RPX1RIUkVTSE9MRCxcbiAgICAgIGV2YWx1YXRpb25QZXJpb2RzOiAxLFxuICAgICAgdGhyZXNob2xkOiAxLFxuICAgICAgdHJlYXRNaXNzaW5nRGF0YTogY2xvdWR3YXRjaC5UcmVhdE1pc3NpbmdEYXRhLklHTk9SRSxcbiAgICB9KVxuXG4gICAgYWxhcm0uYWRkQWxhcm1BY3Rpb24odGhpcy5hY3Rpb24pXG4gICAgYWxhcm0uYWRkT2tBY3Rpb24odGhpcy5hY3Rpb24pXG4gIH1cbn1cbiJdfQ==
|
|
178
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"lambda-alarms.js","sourceRoot":"","sources":["../../src/alarms/lambda-alarms.ts"],"names":[],"mappings":"AAAA,OAAO,KAAK,GAAG,MAAM,aAAa,CAAA;AAClC,OAAO,KAAK,UAAU,MAAM,4BAA4B,CAAA;AAExD,OAAO,KAAK,IAAI,MAAM,sBAAsB,CAAA;AAC5C,OAAO,KAAK,gBAAgB,MAAM,mCAAmC,CAAA;AACrE,OAAO,KAAK,UAAU,MAAM,YAAY,CAAA;AACxC,OAAO,EAAE,sBAAsB,EAAE,MAAM,uBAAuB,CAAA;AAsB9D;;;;;;;;;;;GAWG;AACH,MAAM,OAAO,YAAa,SAAQ,UAAU,CAAC,SAAS;IACnC,WAAW,CAAyB;IACpC,aAAa,CAAyB;IACtC,cAAc,CAAkB;IAChC,UAAU,CAAmB;IAE9C,YACE,KAA2B,EAC3B,EAAU,EACV,KAAwB;QAExB,KAAK,CAAC,KAAK,EAAE,EAAE,CAAC,CAAA;QAEhB,IAAI,CAAC,WAAW,GAAG,KAAK,CAAC,WAAW,CAAA;QACpC,IAAI,CAAC,aAAa,GAAG,KAAK,CAAC,aAAa,CAAA;QACxC,IAAI,CAAC,cAAc,GAAG,KAAK,CAAC,cAAc,CAAA;QAC1C,IAAI,CAAC,UAAU,GAAG,KAAK,CAAC,UAAU,CAAA;IACpC,CAAC;IAED;;;;;OAKG;IACH,uBAAuB,CAAC,KA+BvB;QACC,IAAI,CAAC,wBAAwB,CAAC,KAAK,EAAE,WAAW,CAAC,CAAA;QACjD,IAAI,CAAC,2BAA2B,CAAC,KAAK,EAAE,cAAc,CAAC,CAAA;IACzD,CAAC;IAED;;OAEG;IACH,wBAAwB,CAAC,KAKxB;QACC,IAAI,KAAK,EAAE,OAAO,KAAK,KAAK,EAAE,CAAC;YAC7B,sCAAsC;YACtC,MAAM,MAAM,GAAG,KAAK,EAAE,MAAM,IAAI,IAAI,CAAC,aAAa,CAAA;YAElD,MAAM,KAAK,GAAG,IAAI,UAAU,CAAC,MAAM,CAAC;gBAClC,UAAU,EAAE,QAAQ;gBACpB,SAAS,EAAE,YAAY;gBACvB,SAAS,EAAE,KAAK;gBAChB,MAAM,EAAE,GAAG,CAAC,QAAQ,CAAC,OAAO,CAAC,EAAE,CAAC;gBAChC,aAAa,EAAE;oBACb,YAAY,EAAE,IAAI,CAAC,cAAc,CAAC,YAAY;iBAC/C;aACF,CAAC,CAAC,WAAW,CAAC,IAAI,EAAE,uBAAuB,EAAE;gBAC5C,gBAAgB,EAAE,mBAAmB,IAAI,CAAC,cAAc,CAAC,YAAY,2BAA2B,KAAK,EAAE,wBAAwB,IAAI,EAAE,EAAE;gBACvI,kBAAkB,EAChB,UAAU,CAAC,kBAAkB,CAAC,kCAAkC;gBAClE,iBAAiB,EAAE,CAAC;gBACpB,SAAS,EAAE,CAAC;gBACZ,gBAAgB,EAAE,UAAU,CAAC,gBAAgB,CAAC,MAAM;aACrD,CAAC,CAAA;YAEF,KAAK,CAAC,cAAc,CAAC,MAAM,CAAC,CAAA;YAC5B,IAAI,KAAK,EAAE,aAAa,IAAI,IAAI;gBAAE,KAAK,CAAC,WAAW,CAAC,MAAM,CAAC,CAAA;QAC7D,CAAC;IACH,CAAC;IAED;;;OAGG;IACH,2BAA2B,CAAC,KAoB3B;QACC,IAAI,KAAK,EAAE,OAAO,KAAK,KAAK,EAAE,CAAC;YAC7B,MAAM,KAAK,GAAG,IAAI,UAAU,CAAC,MAAM,CAAC;gBAClC,UAAU,EAAE,QAAQ;gBACpB,SAAS,EAAE,YAAY;gBACvB,SAAS,EAAE,KAAK;gBAChB,MAAM,EAAE,GAAG,CAAC,QAAQ,CAAC,OAAO,CAAC,KAAK,EAAE,aAAa,IAAI,EAAE,CAAC;gBACxD,aAAa,EAAE;oBACb,YAAY,EAAE,IAAI,CAAC,cAAc,CAAC,YAAY;iBAC/C;aACF,CAAC,CAAC,WAAW,CAAC,IAAI,EAAE,uBAAuB,EAAE;gBAC5C,gBAAgB,EAAE,mBAAmB,IAAI,CAAC,cAAc,CAAC,YAAY,wBAAwB,KAAK,EAAE,wBAAwB,IAAI,EAAE,EAAE;gBACpI,kBAAkB,EAChB,UAAU,CAAC,kBAAkB,CAAC,kCAAkC;gBAClE,iBAAiB,EAAE,KAAK,EAAE,iBAAiB,IAAI,CAAC;gBAChD,SAAS,EAAE,KAAK,EAAE,SAAS,IAAI,CAAC;gBAChC,gBAAgB,EAAE,UAAU,CAAC,gBAAgB,CAAC,MAAM;aACrD,CAAC,CAAA;YAEF,mCAAmC;YACnC,MAAM,cAAc,GAAG,KAAK,EAAE,MAAM,IAAI,IAAI,CAAC,WAAW,CAAA;YACxD,KAAK,CAAC,cAAc,CAAC,cAAc,CAAC,CAAA;YACpC,IAAI,KAAK,EAAE,aAAa,IAAI,IAAI,EAAE,CAAC;gBACjC,KAAK,CAAC,WAAW,CAAC,cAAc,CAAC,CAAA;YACnC,CAAC;QACH,CAAC;IACH,CAAC;IAED;;OAEG;IACH,aAAa,CAAC,KAYb;QACC,sEAAsE;QACtE,IAAI,CAAC,IAAI,CAAC,UAAU,EAAE,CAAC;YACrB,MAAM,iBAAiB,GAAG,KAAK,CAAC,QAAQ,CAAC,eAAe,CACtD,mBAAmB,EACnB;gBACE,aAAa,EAAE,sBAAsB,EAAE;gBACvC,UAAU,EAAE,QAAQ;gBACpB,eAAe,EAAE,SAAS,GAAG,CAAC,KAAK,CAAC,EAAE,CAAC,IAAI,CAAC,CAAC,SAAS,IAAI,IAAI,CAAC,cAAc,CAAC,YAAY,SAAS;aACpG,CACF,CAAA;YAED,MAAM,UAAU,GAAG,iBAAiB;iBACjC,MAAM,EAAE;iBACR,IAAI,CAAC;gBACJ,SAAS,EAAE,KAAK;gBAChB,MAAM,EAAE,GAAG,CAAC,QAAQ,CAAC,OAAO,CAAC,EAAE,CAAC;aACjC,CAAC;iBACD,WAAW,CAAC,IAAI,EAAE,eAAe,EAAE;gBAClC,gBAAgB,EACd,KAAK,CAAC,gBAAgB;oBACtB,GAAG,IAAI,CAAC,cAAc,CAAC,YAAY,kBAAkB;gBACvD,iBAAiB,EAAE,CAAC;gBACpB,SAAS,EAAE,CAAC;gBACZ,gBAAgB,EAAE,UAAU,CAAC,gBAAgB,CAAC,aAAa;aAC5D,CAAC,CAAA;YAEJ,sCAAsC;YACtC,MAAM,MAAM,GAAG,KAAK,CAAC,MAAM,IAAI,IAAI,CAAC,aAAa,CAAA;YACjD,UAAU,CAAC,cAAc,CAAC,MAAM,CAAC,CAAA;YACjC,IAAI,KAAK,CAAC,aAAa,IAAI,IAAI,EAAE,CAAC;gBAChC,UAAU,CAAC,WAAW,CAAC,MAAM,CAAC,CAAA;YAChC,CAAC;QACH,CAAC;QAED,IAAI,IAAI,CAAC,UAAU,EAAE,CAAC;YACpB,KAAK,CAAC,QAAQ,CAAC,qBAAqB,CAClC,oDAAoD,EACpD;gBACE,WAAW,EAAE,IAAI,gBAAgB,CAAC,iBAAiB,CAAC,IAAI,CAAC,UAAU,CAAC;gBACpE,aAAa,EAAE,sBAAsB,EAAE;aACxC,CACF,CAAA;QACH,CAAC;IACH,CAAC;IAED;;OAEG;IACH,6BAA6B,CAAC,KAS7B;QACC,IAAI,KAAK,CAAC,OAAO,EAAE,CAAC;YAClB,MAAM,aAAa,GAAG,IAAI,CAAC,aAAa,CAAC,QAAQ,CAAC,qBAAqB,CAAC,CAAA;YAExE,iEAAiE;YACjE,IAAI,CAAC,IAAI,CAAC,UAAU,EAAE,CAAC;gBACrB,MAAM,iBAAiB,GAAG,KAAK,CAAC,QAAQ,CAAC,eAAe,CACtD,6BAA6B,EAC7B;oBACE,aAAa,EAAE,aAAa;oBAC5B,UAAU,EAAE,uBAAuB;oBACnC,eAAe,EAAE,SAAS,GAAG,CAAC,KAAK,CAAC,EAAE,CAAC,IAAI,CAAC,CAAC,SAAS,IAAI,IAAI,CAAC,cAAc,CAAC,YAAY,wBAAwB;iBACnH,CACF,CAAA;gBAED,MAAM,UAAU,GAAG,iBAAiB;qBACjC,MAAM,EAAE;qBACR,IAAI,CAAC;oBACJ,SAAS,EAAE,KAAK;oBAChB,MAAM,EAAE,GAAG,CAAC,QAAQ,CAAC,OAAO,CAAC,EAAE,CAAC;iBACjC,CAAC;qBACD,WAAW,CAAC,IAAI,EAAE,+BAA+B,EAAE;oBAClD,gBAAgB,EACd,KAAK,CAAC,gBAAgB;wBACtB,GAAG,IAAI,CAAC,cAAc,CAAC,YAAY,oCAAoC;oBACzE,iBAAiB,EAAE,CAAC;oBACpB,SAAS,EAAE,CAAC;oBACZ,gBAAgB,EAAE,UAAU,CAAC,gBAAgB,CAAC,aAAa;iBAC5D,CAAC,CAAA;gBAEJ,gCAAgC;gBAChC,MAAM,WAAW,GAAG,KAAK,CAAC,MAAM,IAAI,IAAI,CAAC,aAAa,CAAA;gBACtD,UAAU,CAAC,cAAc,CAAC,WAAW,CAAC,CAAA;gBACtC,IAAI,KAAK,CAAC,aAAa,IAAI,IAAI,EAAE,CAAC;oBAChC,UAAU,CAAC,WAAW,CAAC,WAAW,CAAC,CAAA;gBACrC,CAAC;YACH,CAAC;YAED,+DAA+D;YAC/D,IAAI,IAAI,CAAC,UAAU,EAAE,CAAC;gBACpB,KAAK,CAAC,QAAQ,CAAC,qBAAqB,CAClC,iEAAiE,EACjE;oBACE,WAAW,EAAE,IAAI,gBAAgB,CAAC,iBAAiB,CACjD,IAAI,CAAC,UAAU,CAChB;oBACD,aAAa,EAAE,aAAa;iBAC7B,CACF,CAAA;YACH,CAAC;QACH,CAAC;IACH,CAAC;CACF","sourcesContent":["import * as cdk from \"aws-cdk-lib\"\nimport * as cloudwatch from \"aws-cdk-lib/aws-cloudwatch\"\nimport type * as lambda from \"aws-cdk-lib/aws-lambda\"\nimport * as logs from \"aws-cdk-lib/aws-logs\"\nimport * as logsDestinations from \"aws-cdk-lib/aws-logs-destinations\"\nimport * as constructs from \"constructs\"\nimport { jsonErrorFilterPattern } from \"./log-filter-patterns\"\n\nexport interface LambdaAlarmsProps {\n  /**\n   * The CloudWatch alarm action to use for high-severity alarms (ALARM channel).\n   */\n  alarmAction: cloudwatch.IAlarmAction\n  /**\n   * The CloudWatch alarm action to use for lower-severity warnings (WARNING channel).\n   */\n  warningAction: cloudwatch.IAlarmAction\n  /**\n   * The Lambda to add alarms to.\n   */\n  lambdaFunction: lambda.IFunction\n  /**\n   * Optional Lambda function that will receive forwarded log events.\n   * If provided, subscription filters will be created to forward matching logs\n   */\n  logHandler?: lambda.IFunction\n}\n\n/**\n * Alarms for Lambda functions.\n *\n * Unlike RDS and ECS alarm constructs in this package, `LambdaAlarms` is\n * set up manually by consumers.\n *\n * Defaults:\n * - single invocation (singleError) -> sent to `warningAction` by default\n * - multiple invocation (multipleErrors) -> sent to `alarmAction` by default\n * - error log alarm (ERROR/FATAL JSON logs) -> sent to `warningAction` by default\n * - uncaught Java exception alarm -> sent to `warningAction` by default (disabled by default)\n */\nexport class LambdaAlarms extends constructs.Construct {\n  private readonly alarmAction: cloudwatch.IAlarmAction\n  private readonly warningAction: cloudwatch.IAlarmAction\n  private readonly lambdaFunction: lambda.IFunction\n  private readonly logHandler?: lambda.IFunction\n\n  constructor(\n    scope: constructs.Construct,\n    id: string,\n    props: LambdaAlarmsProps,\n  ) {\n    super(scope, id)\n\n    this.alarmAction = props.alarmAction\n    this.warningAction = props.warningAction\n    this.lambdaFunction = props.lambdaFunction\n    this.logHandler = props.logHandler\n  }\n\n  /**\n   * Creates both the single-invocation warning and the\n   * multiple invocations alarm with reasonable defaults.\n   *\n   * Each created alarm can be overridden via the per-alarm props.\n   */\n  addInvocationErrorAlarm(props?: {\n    singleError?: {\n      appendToAlarmDescription?: string\n      enabled?: boolean\n      action?: cloudwatch.IAlarmAction\n      /**\n       * @default true\n       */\n      enableOkAlarm?: boolean\n    }\n    multipleErrors?: {\n      appendToAlarmDescription?: string\n      enabled?: boolean\n      action?: cloudwatch.IAlarmAction\n      /**\n       * @default true\n       */\n      enableOkAlarm?: boolean\n      /**\n       * @default 3\n       */\n      evaluationPeriods?: number\n      /**\n       * @default 1\n       */\n      threshold?: number\n      /**\n       * @default 60\n       */\n      periodSeconds?: number\n    }\n  }): void {\n    this.addSingleInvocationAlarm(props?.singleError)\n    this.addMultipleInvocationAlarms(props?.multipleErrors)\n  }\n\n  /**\n   * Sets up a CloudWatch Alarm that triggers on a single invocation failure.\n   */\n  addSingleInvocationAlarm(props?: {\n    appendToAlarmDescription?: string\n    enabled?: boolean\n    action?: cloudwatch.IAlarmAction\n    enableOkAlarm?: boolean\n  }): void {\n    if (props?.enabled !== false) {\n      // Sent to warnings channel by default\n      const action = props?.action ?? this.warningAction\n\n      const alarm = new cloudwatch.Metric({\n        metricName: \"Errors\",\n        namespace: \"AWS/Lambda\",\n        statistic: \"Sum\",\n        period: cdk.Duration.seconds(60),\n        dimensionsMap: {\n          FunctionName: this.lambdaFunction.functionName,\n        },\n      }).createAlarm(this, \"SingleInvocationAlarm\", {\n        alarmDescription: `Invocation for '${this.lambdaFunction.functionName}' had a single failure. ${props?.appendToAlarmDescription ?? \"\"}`,\n        comparisonOperator:\n          cloudwatch.ComparisonOperator.GREATER_THAN_OR_EQUAL_TO_THRESHOLD,\n        evaluationPeriods: 1,\n        threshold: 1,\n        treatMissingData: cloudwatch.TreatMissingData.IGNORE,\n      })\n\n      alarm.addAlarmAction(action)\n      if (props?.enableOkAlarm ?? true) alarm.addOkAction(action)\n    }\n  }\n\n  /**\n   * Sets up a CloudWatch Alarm that triggers when there is a short series of\n   * invocation failures.\n   */\n  addMultipleInvocationAlarms(props?: {\n    appendToAlarmDescription?: string\n    enabled?: boolean\n    action?: cloudwatch.IAlarmAction\n    /**\n     * @default true\n     */\n    enableOkAlarm?: boolean\n    /**\n     * @default 3\n     */\n    evaluationPeriods?: number\n    /**\n     * @default 1\n     */\n    threshold?: number\n    /**\n     * @default 60\n     */\n    periodSeconds?: number\n  }): void {\n    if (props?.enabled !== false) {\n      const alarm = new cloudwatch.Metric({\n        metricName: \"Errors\",\n        namespace: \"AWS/Lambda\",\n        statistic: \"Sum\",\n        period: cdk.Duration.seconds(props?.periodSeconds ?? 60),\n        dimensionsMap: {\n          FunctionName: this.lambdaFunction.functionName,\n        },\n      }).createAlarm(this, \"InvocationSeriesAlarm\", {\n        alarmDescription: `Invocation for '${this.lambdaFunction.functionName}' failed repeatedly. ${props?.appendToAlarmDescription ?? \"\"}`,\n        comparisonOperator:\n          cloudwatch.ComparisonOperator.GREATER_THAN_OR_EQUAL_TO_THRESHOLD,\n        evaluationPeriods: props?.evaluationPeriods ?? 3,\n        threshold: props?.threshold ?? 1,\n        treatMissingData: cloudwatch.TreatMissingData.IGNORE,\n      })\n\n      // Sent to alarm channel by default\n      const multipleAction = props?.action ?? this.alarmAction\n      alarm.addAlarmAction(multipleAction)\n      if (props?.enableOkAlarm ?? true) {\n        alarm.addOkAction(multipleAction)\n      }\n    }\n  }\n\n  /**\n   * Alerts when ERROR is logged from lambda.\n   */\n  addErrorAlarm(props: {\n    logGroup: logs.ILogGroup\n    alarmDescription?: string\n    /**\n     * @default true\n     */\n    enableOkAlarm?: boolean\n    /**\n     * An action to use for CloudWatch alarm state changes\n     * instead of the default warning action\n     */\n    action?: cloudwatch.IAlarmAction\n  }): void {\n    // If no log handler is configured, create simple \"ERROR\" metric alarm\n    if (!this.logHandler) {\n      const errorMetricFilter = props.logGroup.addMetricFilter(\n        \"ErrorMetricFilter\",\n        {\n          filterPattern: jsonErrorFilterPattern(),\n          metricName: \"Errors\",\n          metricNamespace: `stack/${cdk.Stack.of(this).stackName}/${this.lambdaFunction.functionName}/Errors`,\n        },\n      )\n\n      const errorAlarm = errorMetricFilter\n        .metric()\n        .with({\n          statistic: \"Sum\",\n          period: cdk.Duration.seconds(60),\n        })\n        .createAlarm(this, \"ErrorLogAlarm\", {\n          alarmDescription:\n            props.alarmDescription ??\n            `${this.lambdaFunction.functionName} logged an error`,\n          evaluationPeriods: 1,\n          threshold: 1,\n          treatMissingData: cloudwatch.TreatMissingData.NOT_BREACHING,\n        })\n\n      // Sent to warnings channel by default\n      const action = props.action ?? this.warningAction\n      errorAlarm.addAlarmAction(action)\n      if (props.enableOkAlarm ?? true) {\n        errorAlarm.addOkAction(action)\n      }\n    }\n\n    if (this.logHandler) {\n      props.logGroup.addSubscriptionFilter(\n        \"liflig-cdk-log-content-to-slack-error-subscription\",\n        {\n          destination: new logsDestinations.LambdaDestination(this.logHandler),\n          filterPattern: jsonErrorFilterPattern(),\n        },\n      )\n    }\n  }\n\n  /**\n   * Adds an alarm for uncaught Java exceptions in the Lambda function's logs.\n   */\n  addUncaughtJavaExceptionAlarm(props: {\n    logGroup: logs.ILogGroup\n    alarmDescription?: string\n    /**\n     * @default false\n     */\n    enabled?: boolean\n    enableOkAlarm?: boolean\n    action?: cloudwatch.IAlarmAction\n  }): void {\n    if (props.enabled) {\n      const filterPattern = logs.FilterPattern.allTerms(\"Exception in thread\")\n\n      // If no log handler is configured, create a simple metric alarm.\n      if (!this.logHandler) {\n        const errorMetricFilter = props.logGroup.addMetricFilter(\n          \"UncaughtJavaExceptionFilter\",\n          {\n            filterPattern: filterPattern,\n            metricName: \"UncaughtJavaException\",\n            metricNamespace: `stack/${cdk.Stack.of(this).stackName}/${this.lambdaFunction.functionName}/UncaughtJavaException`,\n          },\n        )\n\n        const errorAlarm = errorMetricFilter\n          .metric()\n          .with({\n            statistic: \"Sum\",\n            period: cdk.Duration.seconds(60),\n          })\n          .createAlarm(this, \"UncaughtJavaExceptionLogAlarm\", {\n            alarmDescription:\n              props.alarmDescription ??\n              `${this.lambdaFunction.functionName} logged an uncaught Java exception`,\n            evaluationPeriods: 1,\n            threshold: 1,\n            treatMissingData: cloudwatch.TreatMissingData.NOT_BREACHING,\n          })\n\n        // Default to the warning action\n        const actionToUse = props.action ?? this.warningAction\n        errorAlarm.addAlarmAction(actionToUse)\n        if (props.enableOkAlarm ?? true) {\n          errorAlarm.addOkAction(actionToUse)\n        }\n      }\n\n      // If a log handler is configured, forward matching logs to it.\n      if (this.logHandler) {\n        props.logGroup.addSubscriptionFilter(\n          \"liflig-cdk-log-content-to-slack-uncaught-exception-subscription\",\n          {\n            destination: new logsDestinations.LambdaDestination(\n              this.logHandler,\n            ),\n            filterPattern: filterPattern,\n          },\n        )\n      }\n    }\n  }\n}\n"]}
|
|
@@ -0,0 +1,6 @@
|
|
|
1
|
+
import * as logs from "aws-cdk-lib/aws-logs";
|
|
2
|
+
/**
|
|
3
|
+
* Common filter pattern used for JSON log-based error detection.
|
|
4
|
+
*/
|
|
5
|
+
export const jsonErrorFilterPattern = () => logs.FilterPattern.any(logs.FilterPattern.stringValue("$.level", "=", "ERROR"), logs.FilterPattern.stringValue("$.level", "=", "FATAL"), logs.FilterPattern.stringValue("$.requestInfo.status.code", "=", "INTERNAL_SERVER_ERROR"));
|
|
6
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoibG9nLWZpbHRlci1wYXR0ZXJucy5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uLy4uL3NyYy9hbGFybXMvbG9nLWZpbHRlci1wYXR0ZXJucy50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFBQSxPQUFPLEtBQUssSUFBSSxNQUFNLHNCQUFzQixDQUFBO0FBRTVDOztHQUVHO0FBQ0gsTUFBTSxDQUFDLE1BQU0sc0JBQXNCLEdBQUcsR0FBRyxFQUFFLENBQ3pDLElBQUksQ0FBQyxhQUFhLENBQUMsR0FBRyxDQUNwQixJQUFJLENBQUMsYUFBYSxDQUFDLFdBQVcsQ0FBQyxTQUFTLEVBQUUsR0FBRyxFQUFFLE9BQU8sQ0FBQyxFQUN2RCxJQUFJLENBQUMsYUFBYSxDQUFDLFdBQVcsQ0FBQyxTQUFTLEVBQUUsR0FBRyxFQUFFLE9BQU8sQ0FBQyxFQUN2RCxJQUFJLENBQUMsYUFBYSxDQUFDLFdBQVcsQ0FDNUIsMkJBQTJCLEVBQzNCLEdBQUcsRUFDSCx1QkFBdUIsQ0FDeEIsQ0FDRixDQUFBIiwic291cmNlc0NvbnRlbnQiOlsiaW1wb3J0ICogYXMgbG9ncyBmcm9tIFwiYXdzLWNkay1saWIvYXdzLWxvZ3NcIlxuXG4vKipcbiAqIENvbW1vbiBmaWx0ZXIgcGF0dGVybiB1c2VkIGZvciBKU09OIGxvZy1iYXNlZCBlcnJvciBkZXRlY3Rpb24uXG4gKi9cbmV4cG9ydCBjb25zdCBqc29uRXJyb3JGaWx0ZXJQYXR0ZXJuID0gKCkgPT5cbiAgbG9ncy5GaWx0ZXJQYXR0ZXJuLmFueShcbiAgICBsb2dzLkZpbHRlclBhdHRlcm4uc3RyaW5nVmFsdWUoXCIkLmxldmVsXCIsIFwiPVwiLCBcIkVSUk9SXCIpLFxuICAgIGxvZ3MuRmlsdGVyUGF0dGVybi5zdHJpbmdWYWx1ZShcIiQubGV2ZWxcIiwgXCI9XCIsIFwiRkFUQUxcIiksXG4gICAgbG9ncy5GaWx0ZXJQYXR0ZXJuLnN0cmluZ1ZhbHVlKFxuICAgICAgXCIkLnJlcXVlc3RJbmZvLnN0YXR1cy5jb2RlXCIsXG4gICAgICBcIj1cIixcbiAgICAgIFwiSU5URVJOQUxfU0VSVkVSX0VSUk9SXCIsXG4gICAgKSxcbiAgKVxuIl19
|
|
@@ -0,0 +1,88 @@
|
|
|
1
|
+
import * as cdk from "aws-cdk-lib";
|
|
2
|
+
import type { IAlarmAction } from "aws-cdk-lib/aws-cloudwatch";
|
|
3
|
+
import * as cloudwatch from "aws-cdk-lib/aws-cloudwatch";
|
|
4
|
+
import * as constructs from "constructs";
|
|
5
|
+
export interface QueueAlarmsProps {
|
|
6
|
+
alarmAction: cloudwatch.IAlarmAction;
|
|
7
|
+
warningAction: cloudwatch.IAlarmAction;
|
|
8
|
+
queueName: string;
|
|
9
|
+
}
|
|
10
|
+
/**
|
|
11
|
+
* This construct provides a thin wrapper that creates two alarms for
|
|
12
|
+
* SQS queues.
|
|
13
|
+
*
|
|
14
|
+
* Unlike RDS and ECS alarm constructs in this package, `QueueAlarms` is
|
|
15
|
+
* set up manually by consumers (it doesn't auto-wire to resources).
|
|
16
|
+
*
|
|
17
|
+
* Defaults:
|
|
18
|
+
* - Messages-not-being-processed alarm -> sent to `alarmAction` by default
|
|
19
|
+
* - Approximate-age alarm -> sent to `warningAction` by default
|
|
20
|
+
*/
|
|
21
|
+
export declare class QueueAlarms extends constructs.Construct {
|
|
22
|
+
private readonly alarmAction;
|
|
23
|
+
private readonly warningAction;
|
|
24
|
+
private readonly queueName;
|
|
25
|
+
constructor(scope: constructs.Construct, id: string, props: QueueAlarmsProps);
|
|
26
|
+
/**
|
|
27
|
+
* Sets up a CloudWatch Composite Alarm that triggers if messages are not being deleted
|
|
28
|
+
* from queue, and there are visible messages on the queue.
|
|
29
|
+
*/
|
|
30
|
+
addMessagesNotBeingProcessedAlarm(props?: {
|
|
31
|
+
/**
|
|
32
|
+
* Period for metric evaluation as a CDK Duration
|
|
33
|
+
* @default cdk.Duration.seconds(300)
|
|
34
|
+
*/
|
|
35
|
+
period?: cdk.Duration;
|
|
36
|
+
/**
|
|
37
|
+
* Evaluation periods for MessagesVisible metric
|
|
38
|
+
* @default 2
|
|
39
|
+
*/
|
|
40
|
+
evaluationPeriodsMessagesVisible?: number;
|
|
41
|
+
/**
|
|
42
|
+
* Threshold for MessagesVisible metric (minimum)
|
|
43
|
+
* @default 1
|
|
44
|
+
*/
|
|
45
|
+
thresholdMessagesVisible?: number;
|
|
46
|
+
/**
|
|
47
|
+
* Evaluation periods for NumberOfMessagesDeleted metric
|
|
48
|
+
* @default 4
|
|
49
|
+
*/
|
|
50
|
+
evaluationPeriodsMessagesDeleted?: number;
|
|
51
|
+
/**
|
|
52
|
+
* Threshold for NumberOfMessagesDeleted (sum)
|
|
53
|
+
* @default 0
|
|
54
|
+
*/
|
|
55
|
+
thresholdMessagesDeleted?: number;
|
|
56
|
+
/**
|
|
57
|
+
* @default true
|
|
58
|
+
*/
|
|
59
|
+
enableOkAlarm?: boolean;
|
|
60
|
+
/** Per-alarm override of the action to use instead of the construct alarmAction */
|
|
61
|
+
action?: IAlarmAction;
|
|
62
|
+
}): void;
|
|
63
|
+
/**
|
|
64
|
+
* Alerts when the ApproximateAgeOfOldestMessage metric is high.
|
|
65
|
+
*/
|
|
66
|
+
addApproximateAgeOfOldestMessageAlarm(props?: {
|
|
67
|
+
alarmDescription?: string;
|
|
68
|
+
/**
|
|
69
|
+
* @default cdk.Duration.seconds(900) (15 minutes)
|
|
70
|
+
*/
|
|
71
|
+
period?: cdk.Duration;
|
|
72
|
+
/**
|
|
73
|
+
* @default 2
|
|
74
|
+
*/
|
|
75
|
+
evaluationPeriods?: number;
|
|
76
|
+
/**
|
|
77
|
+
* Threshold in seconds for the age of the oldest message
|
|
78
|
+
* @default 900 seconds (15 minutes)
|
|
79
|
+
*/
|
|
80
|
+
thresholdSeconds?: number;
|
|
81
|
+
/**
|
|
82
|
+
* @default true
|
|
83
|
+
*/
|
|
84
|
+
enableOkAlarm?: boolean;
|
|
85
|
+
/** An action to use for CloudWatch alarm state changes instead of the default warningAction */
|
|
86
|
+
action?: IAlarmAction;
|
|
87
|
+
}): void;
|
|
88
|
+
}
|