@rio-cloud/cdk-v2-constructs 6.15.0 → 6.16.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/.jsii +428 -47
- package/docs/API.md +544 -8
- package/docs/changelog.md +19 -0
- package/esbuild.mjs +4 -0
- package/lib/contributions/team-transport-two/pipeline/buildspecs.js +4 -1
- package/lib/contributions/team-transport-two/pipeline/pipeline-stack.js +2 -2
- package/lib/fargate/rio-fargate-service.d.ts +20 -2
- package/lib/fargate/rio-fargate-service.js +31 -11
- package/lib/index.d.ts +2 -0
- package/lib/index.js +4 -2
- package/lib/kafka/kafka-event-spec.d.ts +1 -1
- package/lib/kafka/kafka-event-spec.js +2 -2
- package/lib/kafka/kafka-topic.d.ts +2 -2
- package/lib/kafka/kafka-topic.js +1 -1
- package/lib/ses/helper.d.ts +24 -0
- package/lib/ses/helper.js +44 -0
- package/lib/ses/index.d.ts +1 -0
- package/lib/ses/index.js +18 -0
- package/lib/ses/lambda/ses-logger.cjs +50 -0
- package/lib/ses/lambda/ses-logger.d.ts +2 -0
- package/lib/ses/lambda/ses-logger.js +28 -0
- package/lib/ses/ses-observability.d.ts +63 -0
- package/lib/ses/ses-observability.js +225 -0
- package/package.json +1 -1
- package/version.json +1 -1
package/lib/ses/index.js
ADDED
|
@@ -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,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
package/version.json
CHANGED