@liflig/cdk 3.22.19 → 3.23.0

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
@@ -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 default action to use for CloudWatch alarm state changes.
7
+ * The CloudWatch alarm action to use for high-severity alarms (ALARM channel).
7
8
  */
8
- action: cloudwatch.IAlarmAction;
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
- * By itself, no alarms are created. Use the methods available
18
- * to add alarms.
27
+ * Unlike RDS and ECS alarm constructs in this package, `LambdaAlarms` is
28
+ * set up manually by consumers.
19
29
  *
20
- * Create multiple instances of {@link LambdaAlarms} with different `action`
21
- * if you need an alarm to do multiple things.
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 action;
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
- * Sets up a CloudWatch Alarm that triggers if the Lambda fails invocations.
29
- * This usually happens from uncaught exceptions in the lambda.
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
- * Configuration for an alarm.
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
- * Add extra information to the alarm description, like Runbook URL or steps to triage.
98
+ * @default true
38
99
  */
39
- appendToAlarmDescription?: string;
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
- * By itself, no alarms are created. Use the methods available
8
- * to add alarms.
10
+ * Unlike RDS and ECS alarm constructs in this package, `LambdaAlarms` is
11
+ * set up manually by consumers.
9
12
  *
10
- * Create multiple instances of {@link LambdaAlarms} with different `action`
11
- * if you need an alarm to do multiple things.
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
- action;
20
+ alarmAction;
21
+ warningAction;
15
22
  lambdaFunction;
23
+ logHandler;
16
24
  constructor(scope, id, props) {
17
25
  super(scope, id);
18
- this.action = props.action;
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
- * Sets up a CloudWatch Alarm that triggers if the Lambda fails invocations.
23
- * This usually happens from uncaught exceptions in the lambda.
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
- * Configuration for an alarm.
136
+ * Adds an alarm for uncaught Java exceptions in the Lambda function's logs.
28
137
  */
29
- props) {
30
- const alarm = new cloudwatch.Metric({
31
- metricName: "Errors",
32
- namespace: "AWS/Lambda",
33
- statistic: "Sum",
34
- period: cdk.Duration.seconds(60),
35
- dimensionsMap: {
36
- FunctionName: this.lambdaFunction.functionName,
37
- },
38
- }).createAlarm(this, "FailedInvocationAlarm", {
39
- alarmDescription: `Invocation for '${this.lambdaFunction.functionName}' failed. ${props?.appendToAlarmDescription ?? ""}`,
40
- comparisonOperator: cloudwatch.ComparisonOperator.GREATER_THAN_OR_EQUAL_TO_THRESHOLD,
41
- evaluationPeriods: 1,
42
- threshold: 1,
43
- treatMissingData: cloudwatch.TreatMissingData.IGNORE,
44
- });
45
- alarm.addAlarmAction(this.action);
46
- alarm.addOkAction(this.action);
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":"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","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,5 @@
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 declare const jsonErrorFilterPattern: () => logs.JsonPattern;
@@ -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
+ }