@rio-cloud/cdk-v2-constructs 6.16.2 → 6.16.3

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 CHANGED
@@ -11563,7 +11563,7 @@
11563
11563
  "assembly": "@rio-cloud/cdk-v2-constructs",
11564
11564
  "base": "constructs.Construct",
11565
11565
  "docs": {
11566
- "remarks": "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\nThis construct results in logging all sent emails including the recipient address into CloudWatch logs.\nAs this is considered **PII**, this information MUST be handled accordingly.\nIn 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\nconst rioSesObservability = new RioSesObservability(this, 'ConfigurationSet', {})\nconst 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\nThis is ideal if you want to try out the construct first one some specific emails (e.g. during initial migration).\nThe configuration set name is exported as an SSM parameter `/rio/config/ses-observability/configuration-set-name`.\nThus, you can easily make it available as environment variable in your code.\n\n```kotlin\nimport com.amazonaws.services.simpleemail.model.SendEmailRequest\n\nval request = SendEmailRequest()\n .withConfigurationSetName(SSM_CONFIGURATION_SSM_PARAMETER_VALUE)\n ...\n```\n\n## Automated event processing\n\nIf 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.\nTo enable this the SNS topic ARN is exported via Cloudformation as `RioSesObservabilityTopicArn`.",
11566
+ "remarks": "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\nThis construct results in logging all sent emails including the recipient address into CloudWatch logs.\nAs this is considered **PII**, this information MUST be handled accordingly.\nIn 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\nconst rioSesObservability = new RioSesObservability(this, 'ConfigurationSet', {})\nconst 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\nThis is ideal if you want to try out the construct first one some specific emails (e.g. during initial migration).\nThe configuration set name is exported as an SSM parameter `/rio/config/ses-observability/configuration-set-name`.\nThus, you can easily make it available as environment variable in your code.\n\n```kotlin\nimport com.amazonaws.services.simpleemail.model.SendEmailRequest\n\nval request = SendEmailRequest()\n .withConfigurationSetName(SSM_CONFIGURATION_SSM_PARAMETER_VALUE)\n ...\n```\n\n## Automated event processing\n\nIf 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.\nTo enable this the SNS topic ARN is exported via Cloudformation as `RioSesObservabilityTopicArn`.\n\n# Notes\n## Add necessary permissions for fargate task to enable using ses configuration set\nMake sure you have the following permissions in the task role of the fargate task, e.g.:\n```typescript\n service.taskDefinition.taskRole.addToPrincipalPolicy(\n new iam.PolicyStatement({\n actions: ['ses:sendEmail', 'ses:sendRawEmail'],\n resources: [`arn:aws:ses:<your_region>:<your_account>:configuration-set/<your_configuration_set_name>`],\n }));\n```\n\n## Watchful\nIt might be helpful to add watchful to your stack to\nget notified if the ses-logger lambda fails.",
11567
11567
  "stability": "stable",
11568
11568
  "summary": "RioSesObservability."
11569
11569
  },
@@ -11574,7 +11574,7 @@
11574
11574
  },
11575
11575
  "locationInModule": {
11576
11576
  "filename": "src/ses/ses-observability.ts",
11577
- "line": 74
11577
+ "line": 89
11578
11578
  },
11579
11579
  "parameters": [
11580
11580
  {
@@ -11600,7 +11600,7 @@
11600
11600
  "kind": "class",
11601
11601
  "locationInModule": {
11602
11602
  "filename": "src/ses/ses-observability.ts",
11603
- "line": 67
11603
+ "line": 82
11604
11604
  },
11605
11605
  "name": "SesObservability",
11606
11606
  "properties": [
@@ -11612,7 +11612,7 @@
11612
11612
  "immutable": true,
11613
11613
  "locationInModule": {
11614
11614
  "filename": "src/ses/ses-observability.ts",
11615
- "line": 69
11615
+ "line": 84
11616
11616
  },
11617
11617
  "name": "CONFIGURATION_SET_NAME_SSM_PARAMETER",
11618
11618
  "static": true,
@@ -11628,7 +11628,7 @@
11628
11628
  "immutable": true,
11629
11629
  "locationInModule": {
11630
11630
  "filename": "src/ses/ses-observability.ts",
11631
- "line": 70
11631
+ "line": 85
11632
11632
  },
11633
11633
  "name": "SNS_TOPIC_ARN_EXPORT_NAME",
11634
11634
  "static": true,
@@ -11643,7 +11643,7 @@
11643
11643
  "immutable": true,
11644
11644
  "locationInModule": {
11645
11645
  "filename": "src/ses/ses-observability.ts",
11646
- "line": 72
11646
+ "line": 87
11647
11647
  },
11648
11648
  "name": "configurationSet",
11649
11649
  "type": {
@@ -11657,7 +11657,7 @@
11657
11657
  "immutable": true,
11658
11658
  "locationInModule": {
11659
11659
  "filename": "src/ses/ses-observability.ts",
11660
- "line": 71
11660
+ "line": 86
11661
11661
  },
11662
11662
  "name": "snsTopic",
11663
11663
  "type": {
@@ -20037,7 +20037,7 @@
20037
20037
  "assembly": "@rio-cloud/cdk-v2-constructs",
20038
20038
  "base": "constructs.Construct",
20039
20039
  "docs": {
20040
- "remarks": "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\nThis construct results in logging all sent emails including the recipient address into CloudWatch logs.\nAs this is considered **PII**, this information MUST be handled accordingly.\nIn 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\nconst rioSesObservability = new RioSesObservability(this, 'ConfigurationSet', {})\nconst 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\nThis is ideal if you want to try out the construct first one some specific emails (e.g. during initial migration).\nThe configuration set name is exported as an SSM parameter `/rio/config/ses-observability/configuration-set-name`.\nThus, you can easily make it available as environment variable in your code.\n\n```kotlin\nimport com.amazonaws.services.simpleemail.model.SendEmailRequest\n\nval request = SendEmailRequest()\n .withConfigurationSetName(SSM_CONFIGURATION_SSM_PARAMETER_VALUE)\n ...\n```\n\n## Automated event processing\n\nIf 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.\nTo enable this the SNS topic ARN is exported via Cloudformation as `RioSesObservabilityTopicArn`.",
20040
+ "remarks": "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\nThis construct results in logging all sent emails including the recipient address into CloudWatch logs.\nAs this is considered **PII**, this information MUST be handled accordingly.\nIn 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\nconst rioSesObservability = new RioSesObservability(this, 'ConfigurationSet', {})\nconst 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\nThis is ideal if you want to try out the construct first one some specific emails (e.g. during initial migration).\nThe configuration set name is exported as an SSM parameter `/rio/config/ses-observability/configuration-set-name`.\nThus, you can easily make it available as environment variable in your code.\n\n```kotlin\nimport com.amazonaws.services.simpleemail.model.SendEmailRequest\n\nval request = SendEmailRequest()\n .withConfigurationSetName(SSM_CONFIGURATION_SSM_PARAMETER_VALUE)\n ...\n```\n\n## Automated event processing\n\nIf 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.\nTo enable this the SNS topic ARN is exported via Cloudformation as `RioSesObservabilityTopicArn`.\n\n# Notes\n## Add necessary permissions for fargate task to enable using ses configuration set\nMake sure you have the following permissions in the task role of the fargate task, e.g.:\n```typescript\n service.taskDefinition.taskRole.addToPrincipalPolicy(\n new iam.PolicyStatement({\n actions: ['ses:sendEmail', 'ses:sendRawEmail'],\n resources: [`arn:aws:ses:<your_region>:<your_account>:configuration-set/<your_configuration_set_name>`],\n }));\n```\n\n## Watchful\nIt might be helpful to add watchful to your stack to\nget notified if the ses-logger lambda fails.",
20041
20041
  "stability": "stable",
20042
20042
  "summary": "RioSesObservability."
20043
20043
  },
@@ -20048,7 +20048,7 @@
20048
20048
  },
20049
20049
  "locationInModule": {
20050
20050
  "filename": "src/ses/ses-observability.ts",
20051
- "line": 74
20051
+ "line": 89
20052
20052
  },
20053
20053
  "parameters": [
20054
20054
  {
@@ -20074,7 +20074,7 @@
20074
20074
  "kind": "class",
20075
20075
  "locationInModule": {
20076
20076
  "filename": "src/ses/ses-observability.ts",
20077
- "line": 67
20077
+ "line": 82
20078
20078
  },
20079
20079
  "name": "SesObservability",
20080
20080
  "namespace": "ses",
@@ -20087,7 +20087,7 @@
20087
20087
  "immutable": true,
20088
20088
  "locationInModule": {
20089
20089
  "filename": "src/ses/ses-observability.ts",
20090
- "line": 69
20090
+ "line": 84
20091
20091
  },
20092
20092
  "name": "CONFIGURATION_SET_NAME_SSM_PARAMETER",
20093
20093
  "static": true,
@@ -20103,7 +20103,7 @@
20103
20103
  "immutable": true,
20104
20104
  "locationInModule": {
20105
20105
  "filename": "src/ses/ses-observability.ts",
20106
- "line": 70
20106
+ "line": 85
20107
20107
  },
20108
20108
  "name": "SNS_TOPIC_ARN_EXPORT_NAME",
20109
20109
  "static": true,
@@ -20118,7 +20118,7 @@
20118
20118
  "immutable": true,
20119
20119
  "locationInModule": {
20120
20120
  "filename": "src/ses/ses-observability.ts",
20121
- "line": 72
20121
+ "line": 87
20122
20122
  },
20123
20123
  "name": "configurationSet",
20124
20124
  "type": {
@@ -20132,7 +20132,7 @@
20132
20132
  "immutable": true,
20133
20133
  "locationInModule": {
20134
20134
  "filename": "src/ses/ses-observability.ts",
20135
- "line": 71
20135
+ "line": 86
20136
20136
  },
20137
20137
  "name": "snsTopic",
20138
20138
  "type": {
@@ -20574,5 +20574,5 @@
20574
20574
  }
20575
20575
  },
20576
20576
  "version": "0.0.0",
20577
- "fingerprint": "dE2N/8PIEf6Kfek1j6u+VQPND9PxzEcxYdKNpdiCGQw="
20577
+ "fingerprint": "N7iGFoxgZa8RhCRzeiVg/6/2q7WLz2faRiJrlTzFYpI="
20578
20578
  }
package/docs/API.md CHANGED
@@ -6210,6 +6210,21 @@ If you have specific needs for which you'd like to process the SNS events direct
6210
6210
  (e.g. automatically creating a suppression list from bounces), you can subscribe to the SNS topic.
6211
6211
  To enable this the SNS topic ARN is exported via Cloudformation as `RioSesObservabilityTopicArn`.
6212
6212
 
6213
+ # Notes
6214
+ ## Add necessary permissions for fargate task to enable using ses configuration set
6215
+ Make sure you have the following permissions in the task role of the fargate task, e.g.:
6216
+ ```typescript
6217
+ service.taskDefinition.taskRole.addToPrincipalPolicy(
6218
+ new iam.PolicyStatement({
6219
+ actions: ['ses:sendEmail', 'ses:sendRawEmail'],
6220
+ resources: [`arn:aws:ses:<your_region>:<your_account>:configuration-set/<your_configuration_set_name>`],
6221
+ }));
6222
+ ```
6223
+
6224
+ ## Watchful
6225
+ It might be helpful to add watchful to your stack to
6226
+ get notified if the ses-logger lambda fails.
6227
+
6213
6228
  #### Initializers <a name="Initializers" id="@rio-cloud/cdk-v2-constructs.SesObservability.Initializer"></a>
6214
6229
 
6215
6230
  ```typescript
@@ -6413,6 +6428,21 @@ If you have specific needs for which you'd like to process the SNS events direct
6413
6428
  (e.g. automatically creating a suppression list from bounces), you can subscribe to the SNS topic.
6414
6429
  To enable this the SNS topic ARN is exported via Cloudformation as `RioSesObservabilityTopicArn`.
6415
6430
 
6431
+ # Notes
6432
+ ## Add necessary permissions for fargate task to enable using ses configuration set
6433
+ Make sure you have the following permissions in the task role of the fargate task, e.g.:
6434
+ ```typescript
6435
+ service.taskDefinition.taskRole.addToPrincipalPolicy(
6436
+ new iam.PolicyStatement({
6437
+ actions: ['ses:sendEmail', 'ses:sendRawEmail'],
6438
+ resources: [`arn:aws:ses:<your_region>:<your_account>:configuration-set/<your_configuration_set_name>`],
6439
+ }));
6440
+ ```
6441
+
6442
+ ## Watchful
6443
+ It might be helpful to add watchful to your stack to
6444
+ get notified if the ses-logger lambda fails.
6445
+
6416
6446
  #### Initializers <a name="Initializers" id="@rio-cloud/cdk-v2-constructs.ses.SesObservability.Initializer"></a>
6417
6447
 
6418
6448
  ```typescript
package/docs/changelog.md CHANGED
@@ -2,6 +2,8 @@
2
2
 
3
3
  All notable changes to this project will be documented in this file. See [commit-and-tag-version](https://github.com/absolute-version/commit-and-tag-version) for commit guidelines.
4
4
 
5
+ ## [6.16.3](https://bitbucket.collaboration-man.com/projects/RIODEV/repos/cdk-v2-constructs/compare/commits?targetBranch=refs%2Ftags%2Fv6.16.2&sourceBranch=refs%2Ftags%2Fv6.16.3) (2024-12-10)
6
+
5
7
  ## [6.16.2](https://bitbucket.collaboration-man.com/projects/RIODEV/repos/cdk-v2-constructs/compare/commits?targetBranch=refs%2Ftags%2Fv6.16.1&sourceBranch=refs%2Ftags%2Fv6.16.2) (2024-12-04)
6
8
 
7
9
 
@@ -49,6 +49,21 @@ export interface SesObservabilityProps {
49
49
  * If you have specific needs for which you'd like to process the SNS events directly
50
50
  * (e.g. automatically creating a suppression list from bounces), you can subscribe to the SNS topic.
51
51
  * To enable this the SNS topic ARN is exported via Cloudformation as `RioSesObservabilityTopicArn`.
52
+ *
53
+ * # Notes
54
+ * ## Add necessary permissions for fargate task to enable using ses configuration set
55
+ * Make sure you have the following permissions in the task role of the fargate task, e.g.:
56
+ * ```typescript
57
+ * service.taskDefinition.taskRole.addToPrincipalPolicy(
58
+ * new iam.PolicyStatement({
59
+ * actions: ['ses:sendEmail', 'ses:sendRawEmail'],
60
+ * resources: [`arn:aws:ses:<your_region>:<your_account>:configuration-set/<your_configuration_set_name>`],
61
+ * }));
62
+ * ```
63
+ *
64
+ * ## Watchful
65
+ * It might be helpful to add watchful to your stack to
66
+ * get notified if the ses-logger lambda fails.
52
67
  */
53
68
  export declare class SesObservability extends Construct {
54
69
  static readonly CONFIGURATION_SET_NAME_SSM_PARAMETER = "/rio/config/ses-observability/configuration-set-name";
@@ -56,6 +56,21 @@ const rio_landing_zone_1 = require("../rio-landing-zone");
56
56
  * If you have specific needs for which you'd like to process the SNS events directly
57
57
  * (e.g. automatically creating a suppression list from bounces), you can subscribe to the SNS topic.
58
58
  * To enable this the SNS topic ARN is exported via Cloudformation as `RioSesObservabilityTopicArn`.
59
+ *
60
+ * # Notes
61
+ * ## Add necessary permissions for fargate task to enable using ses configuration set
62
+ * Make sure you have the following permissions in the task role of the fargate task, e.g.:
63
+ * ```typescript
64
+ * service.taskDefinition.taskRole.addToPrincipalPolicy(
65
+ * new iam.PolicyStatement({
66
+ * actions: ['ses:sendEmail', 'ses:sendRawEmail'],
67
+ * resources: [`arn:aws:ses:<your_region>:<your_account>:configuration-set/<your_configuration_set_name>`],
68
+ * }));
69
+ * ```
70
+ *
71
+ * ## Watchful
72
+ * It might be helpful to add watchful to your stack to
73
+ * get notified if the ses-logger lambda fails.
59
74
  */
60
75
  class SesObservability extends constructs_1.Construct {
61
76
  constructor(scope, id, props) {
@@ -99,7 +114,7 @@ class SesObservability extends constructs_1.Construct {
99
114
  value: this.snsTopic.topicArn,
100
115
  exportName: SesObservability.SNS_TOPIC_ARN_EXPORT_NAME,
101
116
  });
102
- const notificationHandle = rio_landing_zone_1.RioLandingZone.getTeamAlertSlackChannelParameter(this);
117
+ const notificationHandle = rio_landing_zone_1.RioLandingZone.getTeamAlertSlackChannelParameter(this).stringValue;
103
118
  new datadogv2_1.DatadogMonitor(this, 'QuotaMonitor', {
104
119
  name: `${accountName} SES quota almost reached`,
105
120
  serviceName: 'aws-ses',
@@ -121,6 +136,7 @@ class SesObservability extends constructs_1.Construct {
121
136
  },
122
137
  evaluationDelay: 900,
123
138
  },
139
+ notification: new datadogv2_1.NoNotification(),
124
140
  });
125
141
  new datadogv2_1.DatadogMonitor(this, 'EmailRejectedMonitor', {
126
142
  name: `${accountName} SES Email Rejected`,
@@ -141,6 +157,7 @@ class SesObservability extends constructs_1.Construct {
141
157
  },
142
158
  evaluationDelay: 900,
143
159
  },
160
+ notification: new datadogv2_1.NoNotification(),
144
161
  });
145
162
  new datadogv2_1.DatadogMonitor(this, 'ComplaintRateMonitor', {
146
163
  name: `${accountName} SES reputation endangered: Complaint rate High`,
@@ -167,6 +184,7 @@ class SesObservability extends constructs_1.Construct {
167
184
  },
168
185
  evaluationDelay: 900,
169
186
  },
187
+ notification: new datadogv2_1.NoNotification(),
170
188
  });
171
189
  new datadogv2_1.DatadogMonitor(this, 'BounceRateMonitor', {
172
190
  name: `${accountName} SES reputation endangered: Bounce Rate High`,
@@ -193,6 +211,7 @@ class SesObservability extends constructs_1.Construct {
193
211
  },
194
212
  evaluationDelay: 900,
195
213
  },
214
+ notification: new datadogv2_1.NoNotification(),
196
215
  });
197
216
  }
198
217
  /**
@@ -222,4 +241,4 @@ SesObservability[_a] = { fqn: "@rio-cloud/cdk-v2-constructs.SesObservability", v
222
241
  // ! changing these values is a breaking change for users
223
242
  SesObservability.CONFIGURATION_SET_NAME_SSM_PARAMETER = '/rio/config/ses-observability/configuration-set-name';
224
243
  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: 'ses-logger.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}"]}
244
+ //# 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, NoNotification } 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 *\n * # Notes\n * ## Add necessary permissions for fargate task to enable using ses configuration set\n * Make sure you have the following permissions in the task role of the fargate task, e.g.:\n * ```typescript\n *     service.taskDefinition.taskRole.addToPrincipalPolicy(\n *       new iam.PolicyStatement({\n *         actions: ['ses:sendEmail', 'ses:sendRawEmail'],\n *         resources: [`arn:aws:ses:<your_region>:<your_account>:configuration-set/<your_configuration_set_name>`],\n *       }));\n * ```\n *\n * ## Watchful\n * It might be helpful to add watchful to your stack to\n * get notified if the ses-logger lambda fails.\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: 'ses-logger.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).stringValue;\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      notification: new NoNotification(),\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      notification: new NoNotification(),\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      notification: new NoNotification(),\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      notification: new NoNotification(),\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.16.2",
18
+ "version": "6.16.3",
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.16.2"
2
+ "version": "6.16.3"
3
3
  }