@rio-cloud/cdk-v2-constructs 6.15.0 → 6.16.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.
@@ -0,0 +1,44 @@
1
+ "use strict";
2
+ Object.defineProperty(exports, "__esModule", { value: true });
3
+ exports.trimIndent = trimIndent;
4
+ /**
5
+ * Trim the indent of a multiline string.
6
+ *
7
+ * This uses the [Tagged Templates](https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Template_literals#tagged_templates) feature of JavaScript.
8
+ *
9
+ * Useful in particular when defining multiline messages for datadog monitors.
10
+ * Usage example:
11
+ *
12
+ * ```typescript
13
+ * const message = trimIndent`
14
+ * {{#is_warning}}
15
+ * The auth-server handles ${props.others.name} slower than usual.
16
+ * {{/is_warning}}
17
+ * `;
18
+ * console.log(message);
19
+ * ```
20
+ * returns the following:
21
+ * ```
22
+ * {{#is_warning}}
23
+ * The auth-server handles ${props.others.name} slower than usual.
24
+ * {{/is_warning}}
25
+ * ```
26
+ */
27
+ function trimIndent(strings, ...values) {
28
+ let rawString = strings.raw.reduce((acc, str, i) => acc + str + (values[i] || ''), '');
29
+ const lines = rawString.split('\n');
30
+ const nonEmptyLines = lines.filter((line) => line.trim().length > 0);
31
+ const indentLengths = nonEmptyLines.map((line) => {
32
+ const match = line.match(/^\s*/);
33
+ if (match === null || match.length === 0) {
34
+ return 0;
35
+ }
36
+ return match[0].length;
37
+ });
38
+ const minIndent = Math.min(...indentLengths);
39
+ return lines
40
+ .map((line) => (line.startsWith(' '.repeat(minIndent)) ? line.slice(minIndent) : line))
41
+ .join('\n')
42
+ .trim();
43
+ }
44
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiaGVscGVyLmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiLi4vLi4vc3JjL3Nlcy9oZWxwZXIudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6Ijs7QUF1QkEsZ0NBaUJDO0FBeENEOzs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7O0dBc0JHO0FBQ0gsU0FBZ0IsVUFBVSxDQUFDLE9BQTZCLEVBQUUsR0FBRyxNQUFhO0lBQ3hFLElBQUksU0FBUyxHQUFHLE9BQU8sQ0FBQyxHQUFHLENBQUMsTUFBTSxDQUFDLENBQUMsR0FBRyxFQUFFLEdBQUcsRUFBRSxDQUFDLEVBQUUsRUFBRSxDQUFDLEdBQUcsR0FBRyxHQUFHLEdBQUcsQ0FBQyxNQUFNLENBQUMsQ0FBQyxDQUFDLElBQUksRUFBRSxDQUFDLEVBQUUsRUFBRSxDQUFDLENBQUM7SUFDdkYsTUFBTSxLQUFLLEdBQUcsU0FBUyxDQUFDLEtBQUssQ0FBQyxJQUFJLENBQUMsQ0FBQztJQUNwQyxNQUFNLGFBQWEsR0FBRyxLQUFLLENBQUMsTUFBTSxDQUFDLENBQUMsSUFBSSxFQUFFLEVBQUUsQ0FBQyxJQUFJLENBQUMsSUFBSSxFQUFFLENBQUMsTUFBTSxHQUFHLENBQUMsQ0FBQyxDQUFDO0lBQ3JFLE1BQU0sYUFBYSxHQUFHLGFBQWEsQ0FBQyxHQUFHLENBQUMsQ0FBQyxJQUFJLEVBQUUsRUFBRTtRQUMvQyxNQUFNLEtBQUssR0FBRyxJQUFJLENBQUMsS0FBSyxDQUFDLE1BQU0sQ0FBQyxDQUFDO1FBQ2pDLElBQUksS0FBSyxLQUFLLElBQUksSUFBSSxLQUFLLENBQUMsTUFBTSxLQUFLLENBQUMsRUFBRSxDQUFDO1lBQ3pDLE9BQU8sQ0FBQyxDQUFDO1FBQ1gsQ0FBQztRQUNELE9BQU8sS0FBSyxDQUFDLENBQUMsQ0FBQyxDQUFDLE1BQU0sQ0FBQztJQUN6QixDQUFDLENBQUMsQ0FBQztJQUNILE1BQU0sU0FBUyxHQUFHLElBQUksQ0FBQyxHQUFHLENBQUMsR0FBRyxhQUFhLENBQUMsQ0FBQztJQUU3QyxPQUFPLEtBQUs7U0FDVCxHQUFHLENBQUMsQ0FBQyxJQUFJLEVBQUUsRUFBRSxDQUFDLENBQUMsSUFBSSxDQUFDLFVBQVUsQ0FBQyxHQUFHLENBQUMsTUFBTSxDQUFDLFNBQVMsQ0FBQyxDQUFDLENBQUMsQ0FBQyxDQUFDLElBQUksQ0FBQyxLQUFLLENBQUMsU0FBUyxDQUFDLENBQUMsQ0FBQyxDQUFDLElBQUksQ0FBQyxDQUFDO1NBQ3RGLElBQUksQ0FBQyxJQUFJLENBQUM7U0FDVixJQUFJLEVBQUUsQ0FBQztBQUNaLENBQUMiLCJzb3VyY2VzQ29udGVudCI6WyIvKipcbiAqIFRyaW0gdGhlIGluZGVudCBvZiBhIG11bHRpbGluZSBzdHJpbmcuXG4gKlxuICogVGhpcyB1c2VzIHRoZSBbVGFnZ2VkIFRlbXBsYXRlc10oaHR0cHM6Ly9kZXZlbG9wZXIubW96aWxsYS5vcmcvZW4tVVMvZG9jcy9XZWIvSmF2YVNjcmlwdC9SZWZlcmVuY2UvVGVtcGxhdGVfbGl0ZXJhbHMjdGFnZ2VkX3RlbXBsYXRlcykgZmVhdHVyZSBvZiBKYXZhU2NyaXB0LlxuICpcbiAqIFVzZWZ1bCBpbiBwYXJ0aWN1bGFyIHdoZW4gZGVmaW5pbmcgbXVsdGlsaW5lIG1lc3NhZ2VzIGZvciBkYXRhZG9nIG1vbml0b3JzLlxuICogVXNhZ2UgZXhhbXBsZTpcbiAqXG4gKiBgYGB0eXBlc2NyaXB0XG4gKiBjb25zdCBtZXNzYWdlID0gdHJpbUluZGVudGBcbiAqICAgICB7eyNpc193YXJuaW5nfX1cbiAqICAgICAgIFRoZSBhdXRoLXNlcnZlciBoYW5kbGVzICR7cHJvcHMub3RoZXJzLm5hbWV9IHNsb3dlciB0aGFuIHVzdWFsLlxuICogICAgIHt7L2lzX3dhcm5pbmd9fVxuICogICBgO1xuICogY29uc29sZS5sb2cobWVzc2FnZSk7XG4gKiBgYGBcbiAqIHJldHVybnMgdGhlIGZvbGxvd2luZzpcbiAqIGBgYFxuICoge3sjaXNfd2FybmluZ319XG4gKiAgIFRoZSBhdXRoLXNlcnZlciBoYW5kbGVzICR7cHJvcHMub3RoZXJzLm5hbWV9IHNsb3dlciB0aGFuIHVzdWFsLlxuICoge3svaXNfd2FybmluZ319XG4gKiBgYGBcbiAqL1xuZXhwb3J0IGZ1bmN0aW9uIHRyaW1JbmRlbnQoc3RyaW5nczogVGVtcGxhdGVTdHJpbmdzQXJyYXksIC4uLnZhbHVlczogYW55W10pOiBzdHJpbmcge1xuICBsZXQgcmF3U3RyaW5nID0gc3RyaW5ncy5yYXcucmVkdWNlKChhY2MsIHN0ciwgaSkgPT4gYWNjICsgc3RyICsgKHZhbHVlc1tpXSB8fCAnJyksICcnKTtcbiAgY29uc3QgbGluZXMgPSByYXdTdHJpbmcuc3BsaXQoJ1xcbicpO1xuICBjb25zdCBub25FbXB0eUxpbmVzID0gbGluZXMuZmlsdGVyKChsaW5lKSA9PiBsaW5lLnRyaW0oKS5sZW5ndGggPiAwKTtcbiAgY29uc3QgaW5kZW50TGVuZ3RocyA9IG5vbkVtcHR5TGluZXMubWFwKChsaW5lKSA9PiB7XG4gICAgY29uc3QgbWF0Y2ggPSBsaW5lLm1hdGNoKC9eXFxzKi8pO1xuICAgIGlmIChtYXRjaCA9PT0gbnVsbCB8fCBtYXRjaC5sZW5ndGggPT09IDApIHtcbiAgICAgIHJldHVybiAwO1xuICAgIH1cbiAgICByZXR1cm4gbWF0Y2hbMF0ubGVuZ3RoO1xuICB9KTtcbiAgY29uc3QgbWluSW5kZW50ID0gTWF0aC5taW4oLi4uaW5kZW50TGVuZ3Rocyk7XG5cbiAgcmV0dXJuIGxpbmVzXG4gICAgLm1hcCgobGluZSkgPT4gKGxpbmUuc3RhcnRzV2l0aCgnICcucmVwZWF0KG1pbkluZGVudCkpID8gbGluZS5zbGljZShtaW5JbmRlbnQpIDogbGluZSkpXG4gICAgLmpvaW4oJ1xcbicpXG4gICAgLnRyaW0oKTtcbn1cbiJdfQ==
@@ -0,0 +1 @@
1
+ export * from './ses-observability';
@@ -0,0 +1,18 @@
1
+ "use strict";
2
+ var __createBinding = (this && this.__createBinding) || (Object.create ? (function(o, m, k, k2) {
3
+ if (k2 === undefined) k2 = k;
4
+ var desc = Object.getOwnPropertyDescriptor(m, k);
5
+ if (!desc || ("get" in desc ? !m.__esModule : desc.writable || desc.configurable)) {
6
+ desc = { enumerable: true, get: function() { return m[k]; } };
7
+ }
8
+ Object.defineProperty(o, k2, desc);
9
+ }) : (function(o, m, k, k2) {
10
+ if (k2 === undefined) k2 = k;
11
+ o[k2] = m[k];
12
+ }));
13
+ var __exportStar = (this && this.__exportStar) || function(m, exports) {
14
+ for (var p in m) if (p !== "default" && !Object.prototype.hasOwnProperty.call(exports, p)) __createBinding(exports, m, p);
15
+ };
16
+ Object.defineProperty(exports, "__esModule", { value: true });
17
+ __exportStar(require("./ses-observability"), exports);
18
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiaW5kZXguanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi9zcmMvc2VzL2luZGV4LnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiI7Ozs7Ozs7Ozs7Ozs7Ozs7QUFBQSxzREFBb0MiLCJzb3VyY2VzQ29udGVudCI6WyJleHBvcnQgKiBmcm9tICcuL3Nlcy1vYnNlcnZhYmlsaXR5JzsiXX0=
@@ -0,0 +1,50 @@
1
+ "use strict";
2
+ var __defProp = Object.defineProperty;
3
+ var __getOwnPropDesc = Object.getOwnPropertyDescriptor;
4
+ var __getOwnPropNames = Object.getOwnPropertyNames;
5
+ var __hasOwnProp = Object.prototype.hasOwnProperty;
6
+ var __export = (target, all) => {
7
+ for (var name in all)
8
+ __defProp(target, name, { get: all[name], enumerable: true });
9
+ };
10
+ var __copyProps = (to, from, except, desc) => {
11
+ if (from && typeof from === "object" || typeof from === "function") {
12
+ for (let key of __getOwnPropNames(from))
13
+ if (!__hasOwnProp.call(to, key) && key !== except)
14
+ __defProp(to, key, { get: () => from[key], enumerable: !(desc = __getOwnPropDesc(from, key)) || desc.enumerable });
15
+ }
16
+ return to;
17
+ };
18
+ var __toCommonJS = (mod) => __copyProps(__defProp({}, "__esModule", { value: true }), mod);
19
+
20
+ // src/ses/lambda/ses-logger.ts
21
+ var ses_logger_exports = {};
22
+ __export(ses_logger_exports, {
23
+ handler: () => handler
24
+ });
25
+ module.exports = __toCommonJS(ses_logger_exports);
26
+ var handler = async (event) => {
27
+ try {
28
+ const perEmailEvents = event.Records.map(({ Sns, ...rest }) => ({
29
+ ...event,
30
+ Records: void 0,
31
+ Record: {
32
+ ...rest,
33
+ Sns: {
34
+ ...Sns,
35
+ Message: JSON.parse(Sns.Message)
36
+ }
37
+ }
38
+ }));
39
+ perEmailEvents.forEach((perEmailEvent) => {
40
+ console.log(perEmailEvent);
41
+ });
42
+ } catch (err) {
43
+ console.error(err);
44
+ console.log(event);
45
+ }
46
+ };
47
+ // Annotate the CommonJS export names for ESM import in node:
48
+ 0 && (module.exports = {
49
+ handler
50
+ });
@@ -0,0 +1,2 @@
1
+ import { SNSEvent } from 'aws-lambda';
2
+ export declare const handler: (event: SNSEvent) => Promise<void>;
@@ -0,0 +1,28 @@
1
+ "use strict";
2
+ Object.defineProperty(exports, "__esModule", { value: true });
3
+ exports.handler = void 0;
4
+ const handler = async (event) => {
5
+ try {
6
+ // Parse and write one log per event/email
7
+ const perEmailEvents = event.Records.map(({ Sns, ...rest }) => ({
8
+ ...event,
9
+ Records: undefined,
10
+ Record: {
11
+ ...rest,
12
+ Sns: {
13
+ ...Sns,
14
+ Message: JSON.parse(Sns.Message),
15
+ },
16
+ },
17
+ }));
18
+ perEmailEvents.forEach((perEmailEvent) => {
19
+ console.log(perEmailEvent);
20
+ });
21
+ }
22
+ catch (err) {
23
+ console.error(err);
24
+ console.log(event);
25
+ }
26
+ };
27
+ exports.handler = handler;
28
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoic2VzLWxvZ2dlci5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uLy4uLy4uL3NyYy9zZXMvbGFtYmRhL3Nlcy1sb2dnZXIudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6Ijs7O0FBRU8sTUFBTSxPQUFPLEdBQUcsS0FBSyxFQUFFLEtBQWUsRUFBRSxFQUFFO0lBQy9DLElBQUksQ0FBQztRQUNILDBDQUEwQztRQUMxQyxNQUFNLGNBQWMsR0FBRyxLQUFLLENBQUMsT0FBTyxDQUFDLEdBQUcsQ0FBQyxDQUFDLEVBQUUsR0FBRyxFQUFFLEdBQUcsSUFBSSxFQUFFLEVBQUUsRUFBRSxDQUFDLENBQUM7WUFDOUQsR0FBRyxLQUFLO1lBQ1IsT0FBTyxFQUFFLFNBQVM7WUFDbEIsTUFBTSxFQUFFO2dCQUNOLEdBQUcsSUFBSTtnQkFDUCxHQUFHLEVBQUU7b0JBQ0gsR0FBRyxHQUFHO29CQUNOLE9BQU8sRUFBRSxJQUFJLENBQUMsS0FBSyxDQUFDLEdBQUcsQ0FBQyxPQUFPLENBQVc7aUJBQzNDO2FBQ0Y7U0FDRixDQUFDLENBQUMsQ0FBQztRQUNKLGNBQWMsQ0FBQyxPQUFPLENBQUMsQ0FBQyxhQUFhLEVBQUUsRUFBRTtZQUN2QyxPQUFPLENBQUMsR0FBRyxDQUFDLGFBQWEsQ0FBQyxDQUFDO1FBQzdCLENBQUMsQ0FBQyxDQUFDO0lBQ0wsQ0FBQztJQUFDLE9BQU8sR0FBRyxFQUFFLENBQUM7UUFDYixPQUFPLENBQUMsS0FBSyxDQUFDLEdBQUcsQ0FBQyxDQUFDO1FBQ25CLE9BQU8sQ0FBQyxHQUFHLENBQUMsS0FBSyxDQUFDLENBQUM7SUFDckIsQ0FBQztBQUNILENBQUMsQ0FBQztBQXJCVyxRQUFBLE9BQU8sV0FxQmxCIiwic291cmNlc0NvbnRlbnQiOlsiaW1wb3J0IHsgU05TRXZlbnQgfSBmcm9tICdhd3MtbGFtYmRhJztcblxuZXhwb3J0IGNvbnN0IGhhbmRsZXIgPSBhc3luYyAoZXZlbnQ6IFNOU0V2ZW50KSA9PiB7XG4gIHRyeSB7XG4gICAgLy8gUGFyc2UgYW5kIHdyaXRlIG9uZSBsb2cgcGVyIGV2ZW50L2VtYWlsXG4gICAgY29uc3QgcGVyRW1haWxFdmVudHMgPSBldmVudC5SZWNvcmRzLm1hcCgoeyBTbnMsIC4uLnJlc3QgfSkgPT4gKHtcbiAgICAgIC4uLmV2ZW50LFxuICAgICAgUmVjb3JkczogdW5kZWZpbmVkLFxuICAgICAgUmVjb3JkOiB7XG4gICAgICAgIC4uLnJlc3QsXG4gICAgICAgIFNuczoge1xuICAgICAgICAgIC4uLlNucyxcbiAgICAgICAgICBNZXNzYWdlOiBKU09OLnBhcnNlKFNucy5NZXNzYWdlKSBhcyBvYmplY3QsXG4gICAgICAgIH0sXG4gICAgICB9LFxuICAgIH0pKTtcbiAgICBwZXJFbWFpbEV2ZW50cy5mb3JFYWNoKChwZXJFbWFpbEV2ZW50KSA9PiB7XG4gICAgICBjb25zb2xlLmxvZyhwZXJFbWFpbEV2ZW50KTtcbiAgICB9KTtcbiAgfSBjYXRjaCAoZXJyKSB7XG4gICAgY29uc29sZS5lcnJvcihlcnIpO1xuICAgIGNvbnNvbGUubG9nKGV2ZW50KTtcbiAgfVxufTtcbiJdfQ==
@@ -0,0 +1,63 @@
1
+ import * as ses from 'aws-cdk-lib/aws-ses';
2
+ import * as sns from 'aws-cdk-lib/aws-sns';
3
+ import { Construct } from 'constructs';
4
+ export interface SesObservabilityProps {
5
+ /**
6
+ * The SES events to log.
7
+ *
8
+ * @default [EmailSendingEvent.DELIVERY, EmailSendingEvent.BOUNCE, EmailSendingEvent.COMPLAINT, EmailSendingEvent.REJECT]
9
+ */
10
+ readonly loggedSesEvents?: ses.EmailSendingEvent[];
11
+ }
12
+ /**
13
+ * RioSesObservability
14
+ *
15
+ * This construct sets up observability for SES as defined by [ADR - SES domain reputation observability](https://confluence.collaboration-man.com/display/MAN/ADR+-+SES+domain+reputation+observability).
16
+ *
17
+ * # Warning
18
+ * This construct results in logging all sent emails including the recipient address into CloudWatch logs.
19
+ * As this is considered **PII**, this information MUST be handled accordingly.
20
+ * In particular, logs **MUST NOT** be sent to further third party services (e.g. Datadog) and the retention period for logs **MUST NOT** be extended.
21
+ *
22
+ * # Usage
23
+ * ## Default Configuration Set for Identity
24
+ *
25
+ * ```typescript
26
+ * const rioSesObservability = new RioSesObservability(this, 'ConfigurationSet', {})
27
+ * const identity = new EmailIdentity(this, 'EmailIdentity', {
28
+ * identity: Identity.domain('example.com'),
29
+ * configurationSet: rioSesObservability.configurationSet,
30
+ * });
31
+ * ```
32
+ *
33
+ * ## Specifically using the configuration set per request
34
+ *
35
+ * This is ideal if you want to try out the construct first one some specific emails (e.g. during initial migration).
36
+ * The configuration set name is exported as an SSM parameter `/rio/config/ses-observability/configuration-set-name`.
37
+ * Thus, you can easily make it available as environment variable in your code.
38
+ *
39
+ * ```kotlin
40
+ * import com.amazonaws.services.simpleemail.model.SendEmailRequest
41
+ *
42
+ * val request = SendEmailRequest()
43
+ * .withConfigurationSetName(SSM_CONFIGURATION_SSM_PARAMETER_VALUE)
44
+ * ...
45
+ * ```
46
+ *
47
+ * ## Automated event processing
48
+ *
49
+ * If you have specific needs for which you'd like to process the SNS events directly
50
+ * (e.g. automatically creating a suppression list from bounces), you can subscribe to the SNS topic.
51
+ * To enable this the SNS topic ARN is exported via Cloudformation as `RioSesObservabilityTopicArn`.
52
+ */
53
+ export declare class SesObservability extends Construct {
54
+ static readonly CONFIGURATION_SET_NAME_SSM_PARAMETER = "/rio/config/ses-observability/configuration-set-name";
55
+ static readonly SNS_TOPIC_ARN_EXPORT_NAME = "RioSesObservabilityTopicArn";
56
+ readonly snsTopic: sns.Topic;
57
+ readonly configurationSet: ses.ConfigurationSet;
58
+ constructor(scope: Construct, id: string, props: SesObservabilityProps);
59
+ /**
60
+ * This function ensures that no PII is leaked to Datadog by adding a subscription filter to the log group of the lambda.
61
+ */
62
+ private ensureNoSubscriptionFiltersOnLambdasLogGroup;
63
+ }
@@ -0,0 +1,225 @@
1
+ "use strict";
2
+ var _a;
3
+ Object.defineProperty(exports, "__esModule", { value: true });
4
+ exports.SesObservability = void 0;
5
+ const JSII_RTTI_SYMBOL_1 = Symbol.for("jsii.rtti");
6
+ const path = require("path");
7
+ const aws_cdk_lib_1 = require("aws-cdk-lib");
8
+ const lambda = require("aws-cdk-lib/aws-lambda");
9
+ const aws_logs_1 = require("aws-cdk-lib/aws-logs");
10
+ const ses = require("aws-cdk-lib/aws-ses");
11
+ const sns = require("aws-cdk-lib/aws-sns");
12
+ const snsSubscriptions = require("aws-cdk-lib/aws-sns-subscriptions");
13
+ const ssm = require("aws-cdk-lib/aws-ssm");
14
+ const cdk = require("aws-cdk-lib/core");
15
+ const constructs_1 = require("constructs");
16
+ const helper_1 = require("./helper");
17
+ const datadogv2_1 = require("../datadogv2");
18
+ const rio_landing_zone_1 = require("../rio-landing-zone");
19
+ /**
20
+ * RioSesObservability
21
+ *
22
+ * This construct sets up observability for SES as defined by [ADR - SES domain reputation observability](https://confluence.collaboration-man.com/display/MAN/ADR+-+SES+domain+reputation+observability).
23
+ *
24
+ * # Warning
25
+ * This construct results in logging all sent emails including the recipient address into CloudWatch logs.
26
+ * As this is considered **PII**, this information MUST be handled accordingly.
27
+ * In particular, logs **MUST NOT** be sent to further third party services (e.g. Datadog) and the retention period for logs **MUST NOT** be extended.
28
+ *
29
+ * # Usage
30
+ * ## Default Configuration Set for Identity
31
+ *
32
+ * ```typescript
33
+ * const rioSesObservability = new RioSesObservability(this, 'ConfigurationSet', {})
34
+ * const identity = new EmailIdentity(this, 'EmailIdentity', {
35
+ * identity: Identity.domain('example.com'),
36
+ * configurationSet: rioSesObservability.configurationSet,
37
+ * });
38
+ * ```
39
+ *
40
+ * ## Specifically using the configuration set per request
41
+ *
42
+ * This is ideal if you want to try out the construct first one some specific emails (e.g. during initial migration).
43
+ * The configuration set name is exported as an SSM parameter `/rio/config/ses-observability/configuration-set-name`.
44
+ * Thus, you can easily make it available as environment variable in your code.
45
+ *
46
+ * ```kotlin
47
+ * import com.amazonaws.services.simpleemail.model.SendEmailRequest
48
+ *
49
+ * val request = SendEmailRequest()
50
+ * .withConfigurationSetName(SSM_CONFIGURATION_SSM_PARAMETER_VALUE)
51
+ * ...
52
+ * ```
53
+ *
54
+ * ## Automated event processing
55
+ *
56
+ * If you have specific needs for which you'd like to process the SNS events directly
57
+ * (e.g. automatically creating a suppression list from bounces), you can subscribe to the SNS topic.
58
+ * To enable this the SNS topic ARN is exported via Cloudformation as `RioSesObservabilityTopicArn`.
59
+ */
60
+ class SesObservability extends constructs_1.Construct {
61
+ constructor(scope, id, props) {
62
+ super(scope, id);
63
+ const accountId = aws_cdk_lib_1.Stack.of(this).account;
64
+ const accountName = rio_landing_zone_1.RioLandingZone.getAccountNameParameter(this);
65
+ const parameter = new ssm.StringParameter(this, 'ConfigurationSetName', {
66
+ parameterName: SesObservability.CONFIGURATION_SET_NAME_SSM_PARAMETER,
67
+ stringValue: 'rio-ses-observability-configuration-set',
68
+ });
69
+ this.snsTopic = new sns.Topic(this, 'SnsTopic', {
70
+ topicName: 'rio-ses-observability-topic',
71
+ });
72
+ this.configurationSet = new ses.ConfigurationSet(this, 'ConfigurationSet', {
73
+ configurationSetName: parameter.stringValue,
74
+ });
75
+ this.configurationSet.addEventDestination('EventDestination', {
76
+ configurationSetEventDestinationName: 'rio-ses-observability-event-destination',
77
+ destination: ses.EventDestination.snsTopic(this.snsTopic),
78
+ events: props.loggedSesEvents ?? [
79
+ ses.EmailSendingEvent.DELIVERY,
80
+ ses.EmailSendingEvent.BOUNCE,
81
+ ses.EmailSendingEvent.COMPLAINT,
82
+ ses.EmailSendingEvent.REJECT,
83
+ ],
84
+ });
85
+ const logFunction = new lambda.Function(this, 'LogFunction', {
86
+ functionName: 'rio-ses-observability-log-function',
87
+ runtime: lambda.Runtime.NODEJS_20_X,
88
+ handler: 'index.handler',
89
+ logRetention: aws_logs_1.RetentionDays.THREE_MONTHS,
90
+ systemLogLevelV2: lambda.SystemLogLevel.WARN,
91
+ loggingFormat: lambda.LoggingFormat.JSON,
92
+ code: lambda.Code.fromAsset(path.join(__dirname, 'lambda'), {
93
+ exclude: ['*.js', '*.d.ts'],
94
+ }),
95
+ });
96
+ this.ensureNoSubscriptionFiltersOnLambdasLogGroup(logFunction);
97
+ this.snsTopic.addSubscription(new snsSubscriptions.LambdaSubscription(logFunction));
98
+ new cdk.CfnOutput(this, 'SnsTopicArn', {
99
+ value: this.snsTopic.topicArn,
100
+ exportName: SesObservability.SNS_TOPIC_ARN_EXPORT_NAME,
101
+ });
102
+ const notificationHandle = rio_landing_zone_1.RioLandingZone.getTeamAlertSlackChannelParameter(this);
103
+ new datadogv2_1.DatadogMonitor(this, 'QuotaMonitor', {
104
+ name: `${accountName} SES quota almost reached`,
105
+ serviceName: 'aws-ses',
106
+ monitorType: datadogv2_1.DatadogMonitorQueryAlertType.QUERY_ALERT,
107
+ query: `max(last_1h):max:aws.ses.sent_last_24_hours{account_id:${accountId}} / max:aws.ses.max_24_hour_send{account_id:${accountId}} > 0.8`,
108
+ message: (0, helper_1.trimIndent) `
109
+ You are almost using up you're maximum email sending quota per 24h.
110
+ Either [request an increase](https://docs.aws.amazon.com/ses/latest/dg/manage-sending-quotas-request-increase.html) for your quota by AWS or reduce the amount of emails sent.
111
+ You can find logs about all mails sent at \`${logFunction.logGroup.logGroupName}\`.
112
+
113
+ {{#is_alert}}@slack-${notificationHandle}{{/is_alert}}
114
+ {{#is_alert_recovery}}@slack-${notificationHandle}{{/is_alert_recovery}}
115
+ `,
116
+ priority: 4,
117
+ optionOverrides: {
118
+ thresholds: {
119
+ critical: 0.8,
120
+ warning: 0.7,
121
+ },
122
+ evaluationDelay: 900,
123
+ },
124
+ });
125
+ new datadogv2_1.DatadogMonitor(this, 'EmailRejectedMonitor', {
126
+ name: `${accountName} SES Email Rejected`,
127
+ serviceName: 'aws-ses',
128
+ monitorType: datadogv2_1.DatadogMonitorQueryAlertType.QUERY_ALERT,
129
+ query: `sum(last_1h):max:aws.ses.rejects{account_id:${accountId}} > 0`,
130
+ message: (0, helper_1.trimIndent) `
131
+ SES [rejected](https://docs.aws.amazon.com/ses/latest/dg/monitor-sending-activity.html) sending at least one email.
132
+ Please investigate the root case, as this is not expected to happen in production ever.
133
+ You can find logs about all mails sent at \`${logFunction.logGroup.logGroupName}\`.
134
+
135
+ {{#is_alert}}@slack-${notificationHandle}{{/is_alert}}
136
+ `,
137
+ priority: 3,
138
+ optionOverrides: {
139
+ thresholds: {
140
+ critical: 0,
141
+ },
142
+ evaluationDelay: 900,
143
+ },
144
+ });
145
+ new datadogv2_1.DatadogMonitor(this, 'ComplaintRateMonitor', {
146
+ name: `${accountName} SES reputation endangered: Complaint rate High`,
147
+ serviceName: 'aws-ses',
148
+ monitorType: datadogv2_1.DatadogMonitorQueryAlertType.QUERY_ALERT,
149
+ query: `max(last_1h):max:aws.ses.reputation_complaint_rate{account_id:${accountId} and not ses:*} > 0.001`,
150
+ message: (0, helper_1.trimIndent) `
151
+ Warning Threshold for Complaint rate {{#is_warning}}almost {{/is_warning}}reached (as defined in AWS).
152
+
153
+ Please reduce the number of complaints received for E-mails sent by your Account.
154
+ If this metric reaches \`0.5%\` the email sending capability of your account will be disabled by AWS.
155
+ You can find logs about all mails sent at \`${logFunction.logGroup.logGroupName}\`.
156
+ See [ADR - SES domain reputation observability](https://confluence.collaboration-man.com/display/MAN/ADR+-+SES+domain+reputation+observability).
157
+
158
+
159
+ {{#is_alert}}@slack-${notificationHandle}{{/is_alert}}
160
+ {{#is_alert_recovery}}@slack-${notificationHandle}{{/is_alert_recovery}}
161
+ `,
162
+ priority: 2,
163
+ optionOverrides: {
164
+ thresholds: {
165
+ critical: 0.001,
166
+ warning: 0.0008,
167
+ },
168
+ evaluationDelay: 900,
169
+ },
170
+ });
171
+ new datadogv2_1.DatadogMonitor(this, 'BounceRateMonitor', {
172
+ name: `${accountName} SES reputation endangered: Bounce Rate High`,
173
+ serviceName: 'aws-ses',
174
+ monitorType: datadogv2_1.DatadogMonitorQueryAlertType.QUERY_ALERT,
175
+ query: `max(last_1h):max:aws.ses.reputation_bounce_rate{account_id:${accountId} and not ses:*} > 0.05`,
176
+ message: (0, helper_1.trimIndent) `
177
+ Warning Threshold for Bounce rate {{#is_warning}}almost {{/is_warning}}reached (as defined in AWS).
178
+
179
+ Please reduce the number of bounces for E-mails sent by your Account.
180
+ If this metric reaches \`10%\` the email sending capability of your account will be disabled by AWS.
181
+ You can find logs about all mails sent at \`${logFunction.logGroup.logGroupName}\`.
182
+ See [ADR - SES domain reputation observability](https://confluence.collaboration-man.com/display/MAN/ADR+-+SES+domain+reputation+observability).
183
+
184
+
185
+ {{#is_alert}}@slack-${notificationHandle}{{/is_alert}}
186
+ {{#is_alert_recovery}}@slack-${notificationHandle}{{/is_alert_recovery}}
187
+ `,
188
+ priority: 2,
189
+ optionOverrides: {
190
+ thresholds: {
191
+ critical: 0.05,
192
+ warning: 0.04,
193
+ },
194
+ evaluationDelay: 900,
195
+ },
196
+ });
197
+ }
198
+ /**
199
+ * This function ensures that no PII is leaked to Datadog by adding a subscription filter to the log group of the lambda.
200
+ */
201
+ ensureNoSubscriptionFiltersOnLambdasLogGroup(logFunction) {
202
+ const stack = cdk.Stack.of(this);
203
+ stack.node.addValidation({
204
+ validate: () => {
205
+ const errors = [];
206
+ stack.node.findAll().forEach(node => {
207
+ if (node instanceof aws_logs_1.CfnSubscriptionFilter) {
208
+ const logGroupName = node.logGroupName;
209
+ if (logGroupName === logFunction.logGroup.logGroupName) {
210
+ errors.push('Subscription filters are not allowed ses-observability lambdas log group');
211
+ }
212
+ }
213
+ });
214
+ return errors;
215
+ },
216
+ });
217
+ }
218
+ }
219
+ exports.SesObservability = SesObservability;
220
+ _a = JSII_RTTI_SYMBOL_1;
221
+ SesObservability[_a] = { fqn: "@rio-cloud/cdk-v2-constructs.SesObservability", version: "0.0.0" };
222
+ // ! changing these values is a breaking change for users
223
+ SesObservability.CONFIGURATION_SET_NAME_SSM_PARAMETER = '/rio/config/ses-observability/configuration-set-name';
224
+ SesObservability.SNS_TOPIC_ARN_EXPORT_NAME = 'RioSesObservabilityTopicArn';
225
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"ses-observability.js","sourceRoot":"","sources":["../../src/ses/ses-observability.ts"],"names":[],"mappings":"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","sourcesContent":["import * as path from 'path';\nimport { Stack } from 'aws-cdk-lib';\nimport * as lambda from 'aws-cdk-lib/aws-lambda';\nimport { NodejsFunction } from 'aws-cdk-lib/aws-lambda-nodejs';\nimport { CfnSubscriptionFilter, RetentionDays } from 'aws-cdk-lib/aws-logs';\nimport * as ses from 'aws-cdk-lib/aws-ses';\nimport * as sns from 'aws-cdk-lib/aws-sns';\nimport * as snsSubscriptions from 'aws-cdk-lib/aws-sns-subscriptions';\nimport * as ssm from 'aws-cdk-lib/aws-ssm';\nimport * as cdk from 'aws-cdk-lib/core';\nimport { Construct } from 'constructs';\nimport { trimIndent } from './helper';\nimport { DatadogMonitor, DatadogMonitorQueryAlertType } from '../datadogv2';\nimport { RioLandingZone } from '../rio-landing-zone';\n\n\nexport interface SesObservabilityProps {\n  /**\n     * The SES events to log.\n     *\n     * @default [EmailSendingEvent.DELIVERY, EmailSendingEvent.BOUNCE, EmailSendingEvent.COMPLAINT, EmailSendingEvent.REJECT]\n     */\n  readonly loggedSesEvents?: ses.EmailSendingEvent[];\n}\n\n/**\n * RioSesObservability\n *\n * This construct sets up observability for SES as defined by [ADR - SES domain reputation observability](https://confluence.collaboration-man.com/display/MAN/ADR+-+SES+domain+reputation+observability).\n *\n * # Warning\n * This construct results in logging all sent emails including the recipient address into CloudWatch logs.\n * As this is considered **PII**, this information MUST be handled accordingly.\n * In particular, logs **MUST NOT** be sent to further third party services (e.g. Datadog) and the retention period for logs **MUST NOT** be extended.\n *\n * # Usage\n * ## Default Configuration Set for Identity\n *\n * ```typescript\n * const rioSesObservability = new RioSesObservability(this, 'ConfigurationSet', {})\n * const identity = new EmailIdentity(this, 'EmailIdentity', {\n *     identity: Identity.domain('example.com'),\n *     configurationSet: rioSesObservability.configurationSet,\n * });\n * ```\n *\n * ## Specifically using the configuration set per request\n *\n * This is ideal if you want to try out the construct first one some specific emails (e.g. during initial migration).\n * The configuration set name is exported as an SSM parameter `/rio/config/ses-observability/configuration-set-name`.\n * Thus, you can easily make it available as environment variable in your code.\n *\n * ```kotlin\n * import com.amazonaws.services.simpleemail.model.SendEmailRequest\n *\n * val request = SendEmailRequest()\n *                 .withConfigurationSetName(SSM_CONFIGURATION_SSM_PARAMETER_VALUE)\n *                 ...\n * ```\n *\n * ## Automated event processing\n *\n * If you have specific needs for which you'd like to process the SNS events directly\n * (e.g. automatically creating a suppression list from bounces), you can subscribe to the SNS topic.\n * To enable this the SNS topic ARN is exported via Cloudformation as `RioSesObservabilityTopicArn`.\n */\nexport class SesObservability extends Construct {\n  // ! changing these values is a breaking change for users\n  static readonly CONFIGURATION_SET_NAME_SSM_PARAMETER = '/rio/config/ses-observability/configuration-set-name';\n  static readonly SNS_TOPIC_ARN_EXPORT_NAME = 'RioSesObservabilityTopicArn';\n  readonly snsTopic: sns.Topic;\n  readonly configurationSet: ses.ConfigurationSet;\n\n  constructor(scope: Construct, id: string, props: SesObservabilityProps) {\n    super(scope, id);\n    const accountId = Stack.of(this).account;\n    const accountName = RioLandingZone.getAccountNameParameter(this);\n\n    const parameter = new ssm.StringParameter(this, 'ConfigurationSetName', {\n      parameterName: SesObservability.CONFIGURATION_SET_NAME_SSM_PARAMETER,\n      stringValue: 'rio-ses-observability-configuration-set',\n    });\n\n    this.snsTopic = new sns.Topic(this, 'SnsTopic', {\n      topicName: 'rio-ses-observability-topic',\n    });\n\n    this.configurationSet = new ses.ConfigurationSet(this, 'ConfigurationSet', {\n      configurationSetName: parameter.stringValue,\n    });\n\n    this.configurationSet.addEventDestination('EventDestination', {\n      configurationSetEventDestinationName: 'rio-ses-observability-event-destination',\n      destination: ses.EventDestination.snsTopic(this.snsTopic),\n      events: props.loggedSesEvents ?? [\n        ses.EmailSendingEvent.DELIVERY,\n        ses.EmailSendingEvent.BOUNCE,\n        ses.EmailSendingEvent.COMPLAINT,\n        ses.EmailSendingEvent.REJECT,\n      ],\n    });\n\n    const logFunction = new lambda.Function(this, 'LogFunction', {\n      functionName: 'rio-ses-observability-log-function',\n      runtime: lambda.Runtime.NODEJS_20_X,\n      handler: 'index.handler',\n      logRetention: RetentionDays.THREE_MONTHS,\n      systemLogLevelV2: lambda.SystemLogLevel.WARN,\n      loggingFormat: lambda.LoggingFormat.JSON,\n      code: lambda.Code.fromAsset(path.join(__dirname, 'lambda'), {\n        exclude: ['*.js', '*.d.ts'],\n      }),\n    });\n    this.ensureNoSubscriptionFiltersOnLambdasLogGroup(logFunction);\n\n    this.snsTopic.addSubscription(new snsSubscriptions.LambdaSubscription(logFunction));\n\n    new cdk.CfnOutput(this, 'SnsTopicArn', {\n      value: this.snsTopic.topicArn,\n      exportName: SesObservability.SNS_TOPIC_ARN_EXPORT_NAME,\n    });\n\n    const notificationHandle = RioLandingZone.getTeamAlertSlackChannelParameter(this);\n\n    new DatadogMonitor(this, 'QuotaMonitor', {\n      name: `${accountName} SES quota almost reached`,\n      serviceName: 'aws-ses',\n      monitorType: DatadogMonitorQueryAlertType.QUERY_ALERT,\n      query: `max(last_1h):max:aws.ses.sent_last_24_hours{account_id:${accountId}} / max:aws.ses.max_24_hour_send{account_id:${accountId}} > 0.8`,\n      message: trimIndent`\n        You are almost using up you're maximum email sending quota per 24h. \n        Either [request an increase](https://docs.aws.amazon.com/ses/latest/dg/manage-sending-quotas-request-increase.html) for your quota by AWS or reduce the amount of emails sent.\n        You can find logs about all mails sent at \\`${logFunction.logGroup.logGroupName}\\`.\n        \n        {{#is_alert}}@slack-${notificationHandle}{{/is_alert}}\n        {{#is_alert_recovery}}@slack-${notificationHandle}{{/is_alert_recovery}}\n        `,\n      priority: 4,\n      optionOverrides: {\n        thresholds: {\n          critical: 0.8,\n          warning: 0.7,\n        },\n        evaluationDelay: 900,\n      },\n    });\n\n    new DatadogMonitor(this, 'EmailRejectedMonitor', {\n      name: `${accountName} SES Email Rejected`,\n      serviceName: 'aws-ses',\n      monitorType: DatadogMonitorQueryAlertType.QUERY_ALERT,\n      query: `sum(last_1h):max:aws.ses.rejects{account_id:${accountId}} > 0`,\n      message: trimIndent`\n        SES [rejected](https://docs.aws.amazon.com/ses/latest/dg/monitor-sending-activity.html) sending at least one email.\n        Please investigate the root case, as this is not expected to happen in production ever.\n        You can find logs about all mails sent at \\`${logFunction.logGroup.logGroupName}\\`.\n        \n        {{#is_alert}}@slack-${notificationHandle}{{/is_alert}}\n        `,\n      priority: 3,\n      optionOverrides: {\n        thresholds: {\n          critical: 0,\n        },\n        evaluationDelay: 900,\n      },\n    });\n\n    new DatadogMonitor(this, 'ComplaintRateMonitor', {\n      name: `${accountName} SES reputation endangered: Complaint rate High`,\n      serviceName: 'aws-ses',\n      monitorType: DatadogMonitorQueryAlertType.QUERY_ALERT,\n      query: `max(last_1h):max:aws.ses.reputation_complaint_rate{account_id:${accountId} and not ses:*} > 0.001`,\n      message: trimIndent`\n        Warning Threshold for Complaint rate {{#is_warning}}almost {{/is_warning}}reached (as defined in AWS).\n        \n        Please reduce the number of complaints received for E-mails sent by your Account.\n        If this metric reaches \\`0.5%\\` the email sending capability of your account will be disabled by AWS.\n        You can find logs about all mails sent at \\`${logFunction.logGroup.logGroupName}\\`.\n        See [ADR - SES domain reputation observability](https://confluence.collaboration-man.com/display/MAN/ADR+-+SES+domain+reputation+observability).\n        \n        \n        {{#is_alert}}@slack-${notificationHandle}{{/is_alert}}\n        {{#is_alert_recovery}}@slack-${notificationHandle}{{/is_alert_recovery}}\n        `,\n      priority: 2,\n      optionOverrides: {\n        thresholds: {\n          critical: 0.001,\n          warning: 0.0008,\n        },\n        evaluationDelay: 900,\n      },\n    });\n\n    new DatadogMonitor(this, 'BounceRateMonitor', {\n      name: `${accountName} SES reputation endangered: Bounce Rate High`,\n      serviceName: 'aws-ses',\n      monitorType: DatadogMonitorQueryAlertType.QUERY_ALERT,\n      query: `max(last_1h):max:aws.ses.reputation_bounce_rate{account_id:${accountId} and not ses:*} > 0.05`,\n      message: trimIndent`\n        Warning Threshold for Bounce rate {{#is_warning}}almost {{/is_warning}}reached (as defined in AWS).\n        \n        Please reduce the number of bounces for E-mails sent by your Account.\n        If this metric reaches \\`10%\\` the email sending capability of your account will be disabled by AWS.\n        You can find logs about all mails sent at \\`${logFunction.logGroup.logGroupName}\\`.\n        See [ADR - SES domain reputation observability](https://confluence.collaboration-man.com/display/MAN/ADR+-+SES+domain+reputation+observability).\n        \n        \n        {{#is_alert}}@slack-${notificationHandle}{{/is_alert}}\n        {{#is_alert_recovery}}@slack-${notificationHandle}{{/is_alert_recovery}}\n        `,\n      priority: 2,\n      optionOverrides: {\n        thresholds: {\n          critical: 0.05,\n          warning: 0.04,\n        },\n        evaluationDelay: 900,\n      },\n    });\n  }\n\n  /**\n   * This function ensures that no PII is leaked to Datadog by adding a subscription filter to the log group of the lambda.\n   */\n  private ensureNoSubscriptionFiltersOnLambdasLogGroup(logFunction: NodejsFunction) {\n    const stack = cdk.Stack.of(this);\n    stack.node.addValidation({\n      validate: () => {\n        const errors: string[] = [];\n        stack.node.findAll().forEach(node => {\n          if (node instanceof CfnSubscriptionFilter) {\n            const logGroupName = node.logGroupName;\n            if (logGroupName === logFunction.logGroup.logGroupName) {\n              errors.push('Subscription filters are not allowed ses-observability lambdas log group');\n            }\n          }\n        });\n        return errors;\n      },\n    });\n  }\n}"]}
package/package.json CHANGED
@@ -15,7 +15,7 @@
15
15
  ],
16
16
  "main": "lib/index.js",
17
17
  "license": "Apache-2.0",
18
- "version": "6.15.0",
18
+ "version": "6.16.0",
19
19
  "types": "lib/index.d.ts",
20
20
  "stability": "stable",
21
21
  "exports": {
package/version.json CHANGED
@@ -1,3 +1,3 @@
1
1
  {
2
- "version": "6.15.0"
2
+ "version": "6.16.0"
3
3
  }