@liflig/cdk 3.24.5 → 3.25.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.
@@ -78,6 +78,30 @@ export declare class QueueAlarms extends constructs.Construct {
78
78
  * @default 900 seconds (15 minutes)
79
79
  */
80
80
  thresholdSeconds?: number;
81
+ /**
82
+ * @default false
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
+ /**
89
+ * Alerts when too many messages exist on the queue.
90
+ */
91
+ addTooManyMessagesExistAlarm(props: {
92
+ /**
93
+ * Maximum number of visible messages before triggering the alarm
94
+ */
95
+ messageAmountLimit: number;
96
+ alarmDescription?: string;
97
+ /**
98
+ * @default cdk.Duration.seconds(300)
99
+ */
100
+ period?: cdk.Duration;
101
+ /**
102
+ * @default 1
103
+ */
104
+ evaluationPeriods?: number;
81
105
  /**
82
106
  * @default true
83
107
  */
@@ -101,9 +101,38 @@ export class QueueAlarms extends constructs.Construct {
101
101
  // Sent to warnings channel by default
102
102
  const action = props?.action ?? this.warningAction;
103
103
  ageAlarm.addAlarmAction(action);
104
- if (props?.enableOkAlarm ?? true) {
104
+ if (props?.enableOkAlarm) {
105
105
  ageAlarm.addOkAction(action);
106
106
  }
107
107
  }
108
+ /**
109
+ * Alerts when too many messages exist on the queue.
110
+ */
111
+ addTooManyMessagesExistAlarm(props) {
112
+ const period = props.period ?? cdk.Duration.seconds(300);
113
+ const evaluationPeriods = props.evaluationPeriods ?? 1;
114
+ const alarm = new cloudwatch.Metric({
115
+ metricName: "ApproximateNumberOfMessagesVisible",
116
+ namespace: "AWS/SQS",
117
+ statistic: "Sum",
118
+ period,
119
+ dimensionsMap: {
120
+ QueueName: this.queueName,
121
+ },
122
+ }).createAlarm(this, `TooManyMessagesExist${props.messageAmountLimit}Alarm`, {
123
+ alarmDescription: props.alarmDescription ??
124
+ `${this.queueName} has too many messages (>${props.messageAmountLimit})`,
125
+ comparisonOperator: cloudwatch.ComparisonOperator.GREATER_THAN_OR_EQUAL_TO_THRESHOLD,
126
+ evaluationPeriods,
127
+ threshold: props.messageAmountLimit,
128
+ treatMissingData: cloudwatch.TreatMissingData.IGNORE,
129
+ });
130
+ // Sent to warnings channel by default
131
+ const action = props.action ?? this.warningAction;
132
+ alarm.addAlarmAction(action);
133
+ if (props.enableOkAlarm ?? true) {
134
+ alarm.addOkAction(action);
135
+ }
136
+ }
108
137
  }
109
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"queue-alarms.js","sourceRoot":"","sources":["../../src/alarms/queue-alarms.ts"],"names":[],"mappings":"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","sourcesContent":["import * as cdk from \"aws-cdk-lib\"\nimport type { IAlarmAction } from \"aws-cdk-lib/aws-cloudwatch\"\nimport * as cloudwatch from \"aws-cdk-lib/aws-cloudwatch\"\nimport * as constructs from \"constructs\"\n\nexport interface QueueAlarmsProps {\n  // Action to use for high-severity alarms\n  alarmAction: cloudwatch.IAlarmAction\n  // Action to use for warnings\n  warningAction: cloudwatch.IAlarmAction\n  queueName: string\n}\n\n/**\n * This construct provides a thin wrapper that creates two alarms for\n * SQS queues.\n *\n * Unlike RDS and ECS alarm constructs in this package, `QueueAlarms` is\n * set up manually by consumers (it doesn't auto-wire to resources).\n *\n * Defaults:\n *  - Messages-not-being-processed alarm -> sent to `alarmAction` by default\n *  - Approximate-age alarm -> sent to `warningAction` by default\n */\nexport class QueueAlarms extends constructs.Construct {\n  private readonly alarmAction: cloudwatch.IAlarmAction\n  private readonly warningAction: cloudwatch.IAlarmAction\n  private readonly queueName: string\n\n  constructor(\n    scope: constructs.Construct,\n    id: string,\n    props: QueueAlarmsProps,\n  ) {\n    super(scope, id)\n\n    this.alarmAction = props.alarmAction\n    this.warningAction = props.warningAction\n    this.queueName = props.queueName\n  }\n  /**\n   * Sets up a CloudWatch Composite Alarm that triggers if messages are not being deleted\n   * from queue, and there are visible messages on the queue.\n   */\n  addMessagesNotBeingProcessedAlarm(props?: {\n    /**\n     * Period for metric evaluation as a CDK Duration\n     * @default cdk.Duration.seconds(300)\n     */\n    period?: cdk.Duration\n    /**\n     * Evaluation periods for MessagesVisible metric\n     * @default 2\n     */\n    evaluationPeriodsMessagesVisible?: number\n    /**\n     * Threshold for MessagesVisible metric (minimum)\n     * @default 1\n     */\n    thresholdMessagesVisible?: number\n    /**\n     * Evaluation periods for NumberOfMessagesDeleted metric\n     * @default 4\n     */\n    evaluationPeriodsMessagesDeleted?: number\n    /**\n     * Threshold for NumberOfMessagesDeleted (sum)\n     * @default 0\n     */\n    thresholdMessagesDeleted?: number\n    /**\n     * @default true\n     */\n    enableOkAlarm?: boolean\n    /** Per-alarm override of the action to use instead of the construct alarmAction */\n    action?: IAlarmAction\n  }): void {\n    const period = props?.period ?? cdk.Duration.seconds(300)\n    const evaluationPeriodsMessagesVisible =\n      props?.evaluationPeriodsMessagesVisible ?? 2\n    const thresholdMessagesVisible = props?.thresholdMessagesVisible ?? 1\n    const evaluationPeriodsMessagesDeleted =\n      props?.evaluationPeriodsMessagesDeleted ?? 4\n    const thresholdMessagesDeleted = props?.thresholdMessagesDeleted ?? 0\n\n    const messagesVisibleAlarm = new cloudwatch.Metric({\n      metricName: \"ApproximateNumberOfMessagesVisible\",\n      namespace: \"AWS/SQS\",\n      statistic: \"Minimum\",\n      period,\n      dimensionsMap: {\n        QueueName: this.queueName,\n      },\n    }).createAlarm(this, \"MessagesVisibleAlarm\", {\n      alarmDescription:\n        \"Service might be unavailable! It has available messages on the SQS queue, but these messages are not being deleted (processed).\",\n      comparisonOperator:\n        cloudwatch.ComparisonOperator.GREATER_THAN_OR_EQUAL_TO_THRESHOLD,\n      evaluationPeriods: evaluationPeriodsMessagesVisible,\n      threshold: thresholdMessagesVisible,\n      treatMissingData: cloudwatch.TreatMissingData.IGNORE,\n    })\n\n    const messagesNotBeingDeletedAlarm = new cloudwatch.Metric({\n      metricName: \"NumberOfMessagesDeleted\",\n      namespace: \"AWS/SQS\",\n      statistic: \"Sum\",\n      period,\n      dimensionsMap: {\n        QueueName: this.queueName,\n      },\n    }).createAlarm(this, \"MessagesNotBeingDeleted\", {\n      alarmDescription:\n        \"Service might be unavailable! It has available messages on the SQS queue, but these messages are not being deleted (processed).\",\n      comparisonOperator:\n        cloudwatch.ComparisonOperator.LESS_THAN_OR_EQUAL_TO_THRESHOLD,\n      evaluationPeriods: evaluationPeriodsMessagesDeleted,\n      threshold: thresholdMessagesDeleted,\n      treatMissingData: cloudwatch.TreatMissingData.IGNORE,\n    })\n\n    const messagesNotBeingProcessedAlarm = new cloudwatch.CompositeAlarm(\n      this,\n      \"MessagesNotBeingProcessedAlarm\",\n      {\n        alarmRule: cloudwatch.AlarmRule.allOf(\n          messagesVisibleAlarm,\n          messagesNotBeingDeletedAlarm,\n        ),\n        actionsEnabled: true,\n        alarmDescription:\n          \"Service might be unavailable! It has available messages on the SQS queue, but these messages are not being deleted (processed).\",\n      },\n    )\n\n    // Sent to alarm channel by default\n    const action = props?.action ?? this.alarmAction\n    messagesNotBeingProcessedAlarm.addAlarmAction(action)\n    if (props?.enableOkAlarm ?? true) {\n      messagesNotBeingProcessedAlarm.addOkAction(action)\n    }\n  }\n\n  /**\n   * Alerts when the ApproximateAgeOfOldestMessage metric is high.\n   */\n  addApproximateAgeOfOldestMessageAlarm(props?: {\n    alarmDescription?: string\n    /**\n     * @default cdk.Duration.seconds(900) (15 minutes)\n     */\n    period?: cdk.Duration\n    /**\n     * @default 2\n     */\n    evaluationPeriods?: number\n    /**\n     * Threshold in seconds for the age of the oldest message\n     * @default 900 seconds (15 minutes)\n     */\n    thresholdSeconds?: number\n    /**\n     * @default true\n     */\n    enableOkAlarm?: boolean\n    /** An action to use for CloudWatch alarm state changes instead of the default warningAction */\n    action?: IAlarmAction\n  }): void {\n    const period = props?.period ?? cdk.Duration.seconds(900)\n    const evaluationPeriods = props?.evaluationPeriods ?? 2\n    const threshold = props?.thresholdSeconds ?? 900\n\n    const ageMetric = new cloudwatch.Metric({\n      metricName: \"ApproximateAgeOfOldestMessage\",\n      namespace: \"AWS/SQS\",\n      statistic: \"Maximum\",\n      period,\n      dimensionsMap: {\n        QueueName: this.queueName,\n      },\n    })\n\n    const ageAlarm = ageMetric.createAlarm(\n      this,\n      \"ApproximateAgeOfOldestMessageAlarm\",\n      {\n        alarmDescription:\n          props?.alarmDescription ??\n          `${this.queueName} has an oldest message older than ${threshold} seconds`,\n        comparisonOperator:\n          cloudwatch.ComparisonOperator.GREATER_THAN_OR_EQUAL_TO_THRESHOLD,\n        evaluationPeriods,\n        threshold,\n        treatMissingData: cloudwatch.TreatMissingData.IGNORE,\n      },\n    )\n\n    // Sent to warnings channel by default\n    const action = props?.action ?? this.warningAction\n    ageAlarm.addAlarmAction(action)\n    if (props?.enableOkAlarm ?? true) {\n      ageAlarm.addOkAction(action)\n    }\n  }\n}\n"]}
138
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"queue-alarms.js","sourceRoot":"","sources":["../../src/alarms/queue-alarms.ts"],"names":[],"mappings":"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","sourcesContent":["import * as cdk from \"aws-cdk-lib\"\nimport type { IAlarmAction } from \"aws-cdk-lib/aws-cloudwatch\"\nimport * as cloudwatch from \"aws-cdk-lib/aws-cloudwatch\"\nimport * as constructs from \"constructs\"\n\nexport interface QueueAlarmsProps {\n  // Action to use for high-severity alarms\n  alarmAction: cloudwatch.IAlarmAction\n  // Action to use for warnings\n  warningAction: cloudwatch.IAlarmAction\n  queueName: string\n}\n\n/**\n * This construct provides a thin wrapper that creates two alarms for\n * SQS queues.\n *\n * Unlike RDS and ECS alarm constructs in this package, `QueueAlarms` is\n * set up manually by consumers (it doesn't auto-wire to resources).\n *\n * Defaults:\n *  - Messages-not-being-processed alarm -> sent to `alarmAction` by default\n *  - Approximate-age alarm -> sent to `warningAction` by default\n */\nexport class QueueAlarms extends constructs.Construct {\n  private readonly alarmAction: cloudwatch.IAlarmAction\n  private readonly warningAction: cloudwatch.IAlarmAction\n  private readonly queueName: string\n\n  constructor(\n    scope: constructs.Construct,\n    id: string,\n    props: QueueAlarmsProps,\n  ) {\n    super(scope, id)\n\n    this.alarmAction = props.alarmAction\n    this.warningAction = props.warningAction\n    this.queueName = props.queueName\n  }\n  /**\n   * Sets up a CloudWatch Composite Alarm that triggers if messages are not being deleted\n   * from queue, and there are visible messages on the queue.\n   */\n  addMessagesNotBeingProcessedAlarm(props?: {\n    /**\n     * Period for metric evaluation as a CDK Duration\n     * @default cdk.Duration.seconds(300)\n     */\n    period?: cdk.Duration\n    /**\n     * Evaluation periods for MessagesVisible metric\n     * @default 2\n     */\n    evaluationPeriodsMessagesVisible?: number\n    /**\n     * Threshold for MessagesVisible metric (minimum)\n     * @default 1\n     */\n    thresholdMessagesVisible?: number\n    /**\n     * Evaluation periods for NumberOfMessagesDeleted metric\n     * @default 4\n     */\n    evaluationPeriodsMessagesDeleted?: number\n    /**\n     * Threshold for NumberOfMessagesDeleted (sum)\n     * @default 0\n     */\n    thresholdMessagesDeleted?: number\n    /**\n     * @default true\n     */\n    enableOkAlarm?: boolean\n    /** Per-alarm override of the action to use instead of the construct alarmAction */\n    action?: IAlarmAction\n  }): void {\n    const period = props?.period ?? cdk.Duration.seconds(300)\n    const evaluationPeriodsMessagesVisible =\n      props?.evaluationPeriodsMessagesVisible ?? 2\n    const thresholdMessagesVisible = props?.thresholdMessagesVisible ?? 1\n    const evaluationPeriodsMessagesDeleted =\n      props?.evaluationPeriodsMessagesDeleted ?? 4\n    const thresholdMessagesDeleted = props?.thresholdMessagesDeleted ?? 0\n\n    const messagesVisibleAlarm = new cloudwatch.Metric({\n      metricName: \"ApproximateNumberOfMessagesVisible\",\n      namespace: \"AWS/SQS\",\n      statistic: \"Minimum\",\n      period,\n      dimensionsMap: {\n        QueueName: this.queueName,\n      },\n    }).createAlarm(this, \"MessagesVisibleAlarm\", {\n      alarmDescription:\n        \"Service might be unavailable! It has available messages on the SQS queue, but these messages are not being deleted (processed).\",\n      comparisonOperator:\n        cloudwatch.ComparisonOperator.GREATER_THAN_OR_EQUAL_TO_THRESHOLD,\n      evaluationPeriods: evaluationPeriodsMessagesVisible,\n      threshold: thresholdMessagesVisible,\n      treatMissingData: cloudwatch.TreatMissingData.IGNORE,\n    })\n\n    const messagesNotBeingDeletedAlarm = new cloudwatch.Metric({\n      metricName: \"NumberOfMessagesDeleted\",\n      namespace: \"AWS/SQS\",\n      statistic: \"Sum\",\n      period,\n      dimensionsMap: {\n        QueueName: this.queueName,\n      },\n    }).createAlarm(this, \"MessagesNotBeingDeleted\", {\n      alarmDescription:\n        \"Service might be unavailable! It has available messages on the SQS queue, but these messages are not being deleted (processed).\",\n      comparisonOperator:\n        cloudwatch.ComparisonOperator.LESS_THAN_OR_EQUAL_TO_THRESHOLD,\n      evaluationPeriods: evaluationPeriodsMessagesDeleted,\n      threshold: thresholdMessagesDeleted,\n      treatMissingData: cloudwatch.TreatMissingData.IGNORE,\n    })\n\n    const messagesNotBeingProcessedAlarm = new cloudwatch.CompositeAlarm(\n      this,\n      \"MessagesNotBeingProcessedAlarm\",\n      {\n        alarmRule: cloudwatch.AlarmRule.allOf(\n          messagesVisibleAlarm,\n          messagesNotBeingDeletedAlarm,\n        ),\n        actionsEnabled: true,\n        alarmDescription:\n          \"Service might be unavailable! It has available messages on the SQS queue, but these messages are not being deleted (processed).\",\n      },\n    )\n\n    // Sent to alarm channel by default\n    const action = props?.action ?? this.alarmAction\n    messagesNotBeingProcessedAlarm.addAlarmAction(action)\n    if (props?.enableOkAlarm ?? true) {\n      messagesNotBeingProcessedAlarm.addOkAction(action)\n    }\n  }\n\n  /**\n   * Alerts when the ApproximateAgeOfOldestMessage metric is high.\n   */\n  addApproximateAgeOfOldestMessageAlarm(props?: {\n    alarmDescription?: string\n    /**\n     * @default cdk.Duration.seconds(900) (15 minutes)\n     */\n    period?: cdk.Duration\n    /**\n     * @default 2\n     */\n    evaluationPeriods?: number\n    /**\n     * Threshold in seconds for the age of the oldest message\n     * @default 900 seconds (15 minutes)\n     */\n    thresholdSeconds?: number\n    /**\n     * @default false\n     */\n    enableOkAlarm?: boolean\n    /** An action to use for CloudWatch alarm state changes instead of the default warningAction */\n    action?: IAlarmAction\n  }): void {\n    const period = props?.period ?? cdk.Duration.seconds(900)\n    const evaluationPeriods = props?.evaluationPeriods ?? 2\n    const threshold = props?.thresholdSeconds ?? 900\n\n    const ageMetric = new cloudwatch.Metric({\n      metricName: \"ApproximateAgeOfOldestMessage\",\n      namespace: \"AWS/SQS\",\n      statistic: \"Maximum\",\n      period,\n      dimensionsMap: {\n        QueueName: this.queueName,\n      },\n    })\n\n    const ageAlarm = ageMetric.createAlarm(\n      this,\n      \"ApproximateAgeOfOldestMessageAlarm\",\n      {\n        alarmDescription:\n          props?.alarmDescription ??\n          `${this.queueName} has an oldest message older than ${threshold} seconds`,\n        comparisonOperator:\n          cloudwatch.ComparisonOperator.GREATER_THAN_OR_EQUAL_TO_THRESHOLD,\n        evaluationPeriods,\n        threshold,\n        treatMissingData: cloudwatch.TreatMissingData.IGNORE,\n      },\n    )\n\n    // Sent to warnings channel by default\n    const action = props?.action ?? this.warningAction\n    ageAlarm.addAlarmAction(action)\n    if (props?.enableOkAlarm) {\n      ageAlarm.addOkAction(action)\n    }\n  }\n\n  /**\n   * Alerts when too many messages exist on the queue.\n   */\n  addTooManyMessagesExistAlarm(props: {\n    /**\n     * Maximum number of visible messages before triggering the alarm\n     */\n    messageAmountLimit: number\n    alarmDescription?: string\n    /**\n     * @default cdk.Duration.seconds(300)\n     */\n    period?: cdk.Duration\n    /**\n     * @default 1\n     */\n    evaluationPeriods?: number\n    /**\n     * @default true\n     */\n    enableOkAlarm?: boolean\n    /** An action to use for CloudWatch alarm state changes instead of the default warningAction */\n    action?: IAlarmAction\n  }): void {\n    const period = props.period ?? cdk.Duration.seconds(300)\n    const evaluationPeriods = props.evaluationPeriods ?? 1\n\n    const alarm = new cloudwatch.Metric({\n      metricName: \"ApproximateNumberOfMessagesVisible\",\n      namespace: \"AWS/SQS\",\n      statistic: \"Sum\",\n      period,\n      dimensionsMap: {\n        QueueName: this.queueName,\n      },\n    }).createAlarm(\n      this,\n      `TooManyMessagesExist${props.messageAmountLimit}Alarm`,\n      {\n        alarmDescription:\n          props.alarmDescription ??\n          `${this.queueName} has too many messages (>${props.messageAmountLimit})`,\n        comparisonOperator:\n          cloudwatch.ComparisonOperator.GREATER_THAN_OR_EQUAL_TO_THRESHOLD,\n        evaluationPeriods,\n        threshold: props.messageAmountLimit,\n        treatMissingData: cloudwatch.TreatMissingData.IGNORE,\n      },\n    )\n\n    // Sent to warnings channel by default\n    const action = props.action ?? this.warningAction\n    alarm.addAlarmAction(action)\n    if (props.enableOkAlarm ?? true) {\n      alarm.addOkAction(action)\n    }\n  }\n}\n"]}
package/package.json CHANGED
@@ -1,6 +1,6 @@
1
1
  {
2
2
  "name": "@liflig/cdk",
3
- "version": "3.24.5",
3
+ "version": "3.25.0",
4
4
  "description": "CDK library for Liflig",
5
5
  "type": "module",
6
6
  "repository": {
@@ -51,7 +51,7 @@
51
51
  "@aws-sdk/client-sfn": "3.1030.0",
52
52
  "@aws-sdk/client-ssm": "3.1030.0",
53
53
  "@aws-sdk/lib-storage": "3.1030.0",
54
- "@biomejs/biome": "2.4.11",
54
+ "@biomejs/biome": "2.4.12",
55
55
  "@commitlint/cli": "20.5.0",
56
56
  "@commitlint/config-conventional": "20.5.0",
57
57
  "@types/aws-lambda": "8.10.161",
@@ -63,7 +63,7 @@
63
63
  "esbuild": "0.28.0",
64
64
  "jest": "30.3.0",
65
65
  "jest-cdk-snapshot": "2.3.6",
66
- "lefthook": "2.1.5",
66
+ "lefthook": "2.1.6",
67
67
  "npm-check-updates": "20.0.2",
68
68
  "semantic-release": "25.0.3",
69
69
  "ts-jest": "29.4.9",