@backstage/plugin-notifications-backend-module-email 0.3.10 → 0.3.11-next.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/CHANGELOG.md
CHANGED
|
@@ -1,5 +1,25 @@
|
|
|
1
1
|
# @backstage/plugin-notifications-backend-module-email
|
|
2
2
|
|
|
3
|
+
## 0.3.11-next.1
|
|
4
|
+
|
|
5
|
+
### Patch Changes
|
|
6
|
+
|
|
7
|
+
- f92c9fc: Add optional config for `ses` mail options with `sourceArn`, `fromArn`, `configurationSetName`
|
|
8
|
+
|
|
9
|
+
## 0.3.11-next.0
|
|
10
|
+
|
|
11
|
+
### Patch Changes
|
|
12
|
+
|
|
13
|
+
- Updated dependencies
|
|
14
|
+
- @backstage/config@1.3.3-next.0
|
|
15
|
+
- @backstage/catalog-model@1.7.5-next.0
|
|
16
|
+
- @backstage/catalog-client@1.10.2-next.0
|
|
17
|
+
- @backstage/integration-aws-node@0.1.17-next.0
|
|
18
|
+
- @backstage/backend-plugin-api@1.4.1-next.0
|
|
19
|
+
- @backstage/plugin-notifications-common@0.0.10-next.0
|
|
20
|
+
- @backstage/plugin-catalog-node@1.17.2-next.0
|
|
21
|
+
- @backstage/plugin-notifications-node@0.2.17-next.0
|
|
22
|
+
|
|
3
23
|
## 0.3.10
|
|
4
24
|
|
|
5
25
|
### Patch Changes
|
package/README.md
CHANGED
|
@@ -79,6 +79,11 @@ notifications:
|
|
|
79
79
|
# Who to send email for broadcast notifications
|
|
80
80
|
broadcastConfig:
|
|
81
81
|
receiver: 'users'
|
|
82
|
+
# Optional SES config
|
|
83
|
+
# sesConfig:
|
|
84
|
+
# sourceArn: 'arn:aws:ses:us-west-2:123456789012:identity/example.com'
|
|
85
|
+
# fromArn: 'arn:aws:ses:us-west-2:123456789012:identity/example.com'
|
|
86
|
+
# configurationSetName: 'custom-config'
|
|
82
87
|
# How many emails to send concurrently, defaults to 2
|
|
83
88
|
concurrencyLimit: 10
|
|
84
89
|
# How much to throttle between emails, defaults to 100ms
|
package/config.d.ts
CHANGED
|
@@ -132,6 +132,23 @@ export interface Config {
|
|
|
132
132
|
*/
|
|
133
133
|
receiverEmails?: string[];
|
|
134
134
|
};
|
|
135
|
+
/**
|
|
136
|
+
* Optional SES config for mail options. Allows for delegated sender
|
|
137
|
+
*/
|
|
138
|
+
sesConfig?: {
|
|
139
|
+
/**
|
|
140
|
+
* ARN of the identity to use as the source of the email
|
|
141
|
+
*/
|
|
142
|
+
sourceArn?: string;
|
|
143
|
+
/**
|
|
144
|
+
* ARN of the identity to use for the "From"/sender address of the email
|
|
145
|
+
*/
|
|
146
|
+
fromArn?: string;
|
|
147
|
+
/**
|
|
148
|
+
* Name of the configuration set to use when sending email via ses
|
|
149
|
+
*/
|
|
150
|
+
configurationSetName?: string;
|
|
151
|
+
};
|
|
135
152
|
cache?: {
|
|
136
153
|
/**
|
|
137
154
|
* Email cache TTL, defaults to 1 hour
|
|
@@ -32,6 +32,7 @@ class NotificationsEmailProcessor {
|
|
|
32
32
|
this.broadcastConfig = emailProcessorConfig.getOptionalConfig("broadcastConfig");
|
|
33
33
|
this.sender = emailProcessorConfig.getString("sender");
|
|
34
34
|
this.replyTo = emailProcessorConfig.getOptionalString("replyTo");
|
|
35
|
+
this.sesConfig = emailProcessorConfig.getOptionalConfig("sesConfig");
|
|
35
36
|
this.concurrencyLimit = emailProcessorConfig.getOptionalNumber("concurrencyLimit") ?? 2;
|
|
36
37
|
this.throttleInterval = emailProcessorConfig.has("throttleInterval") ? types.durationToMilliseconds(
|
|
37
38
|
config.readDurationFromConfig(emailProcessorConfig, {
|
|
@@ -55,6 +56,7 @@ class NotificationsEmailProcessor {
|
|
|
55
56
|
transportConfig;
|
|
56
57
|
sender;
|
|
57
58
|
replyTo;
|
|
59
|
+
sesConfig;
|
|
58
60
|
cacheTtl;
|
|
59
61
|
concurrencyLimit;
|
|
60
62
|
throttleInterval;
|
|
@@ -225,13 +227,29 @@ class NotificationsEmailProcessor {
|
|
|
225
227
|
contentParts.push(this.getNotificationLink(notification));
|
|
226
228
|
return contentParts.join("\n\n");
|
|
227
229
|
}
|
|
230
|
+
async getSesOptions() {
|
|
231
|
+
if (!this.sesConfig) {
|
|
232
|
+
return void 0;
|
|
233
|
+
}
|
|
234
|
+
const ses = {};
|
|
235
|
+
const sourceArn = this.sesConfig.getOptionalString("sourceArn");
|
|
236
|
+
const fromArn = this.sesConfig.getOptionalString("fromArn");
|
|
237
|
+
const configurationSetName = this.sesConfig.getOptionalString(
|
|
238
|
+
"configurationSetName"
|
|
239
|
+
);
|
|
240
|
+
if (sourceArn) ses.SourceArn = sourceArn;
|
|
241
|
+
if (fromArn) ses.FromArn = fromArn;
|
|
242
|
+
if (configurationSetName) ses.ConfigurationSetName = configurationSetName;
|
|
243
|
+
return Object.keys(ses).length > 0 ? ses : void 0;
|
|
244
|
+
}
|
|
228
245
|
async sendPlainEmail(notification, emails) {
|
|
229
246
|
const mailOptions = {
|
|
230
247
|
from: this.sender,
|
|
231
248
|
subject: notification.payload.title,
|
|
232
249
|
html: this.getHtmlContent(notification),
|
|
233
250
|
text: this.getTextContent(notification),
|
|
234
|
-
replyTo: this.replyTo
|
|
251
|
+
replyTo: this.replyTo,
|
|
252
|
+
ses: await this.getSesOptions()
|
|
235
253
|
};
|
|
236
254
|
await this.sendMails(mailOptions, emails);
|
|
237
255
|
}
|
|
@@ -241,7 +259,8 @@ class NotificationsEmailProcessor {
|
|
|
241
259
|
subject: await this.templateRenderer?.getSubject?.(notification) ?? notification.payload.title,
|
|
242
260
|
html: await this.templateRenderer?.getHtml?.(notification),
|
|
243
261
|
text: await this.templateRenderer?.getText?.(notification),
|
|
244
|
-
replyTo: this.replyTo
|
|
262
|
+
replyTo: this.replyTo,
|
|
263
|
+
ses: await this.getSesOptions()
|
|
245
264
|
};
|
|
246
265
|
await this.sendMails(mailOptions, emails);
|
|
247
266
|
}
|
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"NotificationsEmailProcessor.cjs.js","sources":["../../src/processor/NotificationsEmailProcessor.ts"],"sourcesContent":["/*\n * Copyright 2024 The Backstage Authors\n *\n * Licensed under the Apache License, Version 2.0 (the \"License\");\n * you may not use this file except in compliance with the License.\n * You may obtain a copy of the License at\n *\n * http://www.apache.org/licenses/LICENSE-2.0\n *\n * Unless required by applicable law or agreed to in writing, software\n * distributed under the License is distributed on an \"AS IS\" BASIS,\n * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.\n * See the License for the specific language governing permissions and\n * limitations under the License.\n */\n\nimport {\n NotificationProcessor,\n NotificationSendOptions,\n} from '@backstage/plugin-notifications-node';\nimport {\n AuthService,\n CacheService,\n LoggerService,\n} from '@backstage/backend-plugin-api';\nimport { Config, readDurationFromConfig } from '@backstage/config';\nimport { durationToMilliseconds } from '@backstage/types';\nimport { CATALOG_FILTER_EXISTS } from '@backstage/catalog-client';\nimport {\n getProcessorFiltersFromConfig,\n Notification,\n NotificationProcessorFilters,\n} from '@backstage/plugin-notifications-common';\nimport {\n createAzureTransport,\n createSendmailTransport,\n createSesTransport,\n createSmtpTransport,\n createStreamTransport,\n} from './transports';\nimport { UserEntity } from '@backstage/catalog-model';\nimport { CatalogService } from '@backstage/plugin-catalog-node';\nimport { compact } from 'lodash';\nimport { DefaultAwsCredentialsManager } from '@backstage/integration-aws-node';\nimport { NotificationTemplateRenderer } from '../extensions';\nimport Mail from 'nodemailer/lib/mailer';\nimport pThrottle from 'p-throttle';\n\nexport class NotificationsEmailProcessor implements NotificationProcessor {\n private transporter: any;\n private readonly broadcastConfig?: Config;\n private readonly transportConfig: Config;\n private readonly sender: string;\n private readonly replyTo?: string;\n private readonly cacheTtl: number;\n private readonly concurrencyLimit: number;\n private readonly throttleInterval: number;\n private readonly frontendBaseUrl: string;\n private readonly filter: NotificationProcessorFilters;\n private readonly allowlistEmailAddresses?: string[];\n private readonly denylistEmailAddresses?: string[];\n\n constructor(\n private readonly logger: LoggerService,\n private readonly config: Config,\n private readonly catalog: CatalogService,\n private readonly auth: AuthService,\n private readonly cache?: CacheService,\n private readonly templateRenderer?: NotificationTemplateRenderer,\n ) {\n const emailProcessorConfig = config.getConfig(\n 'notifications.processors.email',\n );\n this.transportConfig = emailProcessorConfig.getConfig('transportConfig');\n this.broadcastConfig =\n emailProcessorConfig.getOptionalConfig('broadcastConfig');\n this.sender = emailProcessorConfig.getString('sender');\n this.replyTo = emailProcessorConfig.getOptionalString('replyTo');\n this.concurrencyLimit =\n emailProcessorConfig.getOptionalNumber('concurrencyLimit') ?? 2;\n this.throttleInterval = emailProcessorConfig.has('throttleInterval')\n ? durationToMilliseconds(\n readDurationFromConfig(emailProcessorConfig, {\n key: 'throttleInterval',\n }),\n )\n : 100;\n this.cacheTtl = emailProcessorConfig.has('cache.ttl')\n ? durationToMilliseconds(\n readDurationFromConfig(emailProcessorConfig, { key: 'cache.ttl' }),\n )\n : 3_600_000;\n this.frontendBaseUrl = config.getString('app.baseUrl');\n this.allowlistEmailAddresses = emailProcessorConfig.getOptionalStringArray(\n 'allowlistEmailAddresses',\n );\n this.denylistEmailAddresses = emailProcessorConfig.getOptionalStringArray(\n 'denylistEmailAddresses',\n );\n this.filter = getProcessorFiltersFromConfig(emailProcessorConfig);\n }\n\n private async getTransporter() {\n if (this.transporter) {\n return this.transporter;\n }\n const transport = this.transportConfig.getString('transport');\n if (transport === 'smtp') {\n this.transporter = createSmtpTransport(this.transportConfig);\n } else if (transport === 'ses') {\n const awsCredentialsManager = DefaultAwsCredentialsManager.fromConfig(\n this.config,\n );\n this.transporter = await createSesTransport(\n this.transportConfig,\n awsCredentialsManager,\n );\n } else if (transport === 'sendmail') {\n this.transporter = createSendmailTransport(this.transportConfig);\n } else if (transport === 'stream') {\n this.transporter = createStreamTransport();\n } else if (transport === 'azure') {\n this.transporter = createAzureTransport(this.transportConfig);\n } else {\n throw new Error(`Unsupported transport: ${transport}`);\n }\n return this.transporter;\n }\n\n getName(): string {\n return 'Email';\n }\n\n private async getBroadcastEmails(): Promise<string[]> {\n if (!this.broadcastConfig) {\n return [];\n }\n\n const receiver = this.broadcastConfig.getString('receiver');\n if (receiver === 'none') {\n return [];\n }\n\n if (receiver === 'config') {\n return (\n this.broadcastConfig.getOptionalStringArray('receiverEmails') ?? []\n );\n }\n\n if (receiver === 'users') {\n const cached = await this.cache?.get<string[]>('user-emails:all');\n if (cached) {\n return cached;\n }\n\n const entities = await this.catalog.getEntities(\n {\n filter: [\n { kind: 'user', 'spec.profile.email': CATALOG_FILTER_EXISTS },\n ],\n fields: ['spec.profile.email'],\n },\n { credentials: await this.auth.getOwnServiceCredentials() },\n );\n const ret = compact([\n ...new Set(\n entities.items.map(entity => {\n return (entity as UserEntity)?.spec.profile?.email;\n }),\n ),\n ]);\n\n await this.cache?.set('user-emails:all', ret, {\n ttl: this.cacheTtl,\n });\n return ret;\n }\n\n throw new Error(`Unsupported broadcast receiver: ${receiver}`);\n }\n\n private async getUserEmail(entityRef: string): Promise<string[]> {\n const cached = await this.cache?.get<string[]>(`user-emails:${entityRef}`);\n if (cached) {\n return cached;\n }\n\n const entity = await this.catalog.getEntityByRef(entityRef, {\n credentials: await this.auth.getOwnServiceCredentials(),\n });\n const ret: string[] = [];\n if (entity) {\n const userEntity = entity as UserEntity;\n if (userEntity.spec.profile?.email) {\n ret.push(userEntity.spec.profile.email);\n }\n }\n\n await this.cache?.set(`user-emails:${entityRef}`, ret, {\n ttl: this.cacheTtl,\n });\n\n return ret;\n }\n\n private async getRecipientEmails(\n notification: Notification,\n options: NotificationSendOptions,\n ): Promise<string[]> {\n let emails: string[];\n if (options.recipients.type === 'broadcast') {\n emails = await this.getBroadcastEmails();\n } else if (options.recipients.type === 'entity' && !!notification.user) {\n emails = await this.getUserEmail(notification.user);\n } else {\n this.logger.info(\n `Unknown notification type ${options.recipients.type} or missing user.`,\n );\n return [];\n }\n\n if (this.allowlistEmailAddresses) {\n emails = emails.filter(email =>\n this.allowlistEmailAddresses?.includes(email),\n );\n }\n\n if (this.denylistEmailAddresses) {\n emails = emails.filter(\n email => !this.denylistEmailAddresses?.includes(email),\n );\n }\n return emails;\n }\n\n private async sendMail(options: Mail.Options) {\n try {\n this.logger.debug(`Sending notification email to ${options.to}`);\n await this.transporter.sendMail(options);\n } catch (e) {\n this.logger.error(`Failed to send email to ${options.to}: ${e}`);\n }\n }\n\n private async sendMails(options: Mail.Options, emails: string[]) {\n const throttle = pThrottle({\n limit: this.concurrencyLimit,\n interval: this.throttleInterval,\n });\n\n const throttled = throttle((opts: Mail.Options) => this.sendMail(opts));\n await Promise.all(\n emails.map(email => throttled({ ...options, to: email })),\n );\n }\n\n private getNotificationLink(notification: Notification) {\n if (notification.payload.link) {\n const stripLeadingSlash = (s: string) => s.replace(/^\\//, '');\n const ensureTrailingSlash = (s: string) => s.replace(/\\/?$/, '/');\n\n try {\n const url = new URL(\n stripLeadingSlash(notification.payload.link),\n ensureTrailingSlash(this.frontendBaseUrl),\n );\n return url.toString();\n } catch (_e) {\n // noop: fallback to relative URL\n }\n return notification.payload.link;\n }\n return `${this.frontendBaseUrl}/notifications`;\n }\n\n private getHtmlContent(notification: Notification) {\n const contentParts: string[] = [];\n if (notification.payload.description) {\n contentParts.push(`${notification.payload.description}`);\n }\n const link = this.getNotificationLink(notification);\n contentParts.push(`<a href=\"${link}\">${link}</a>`);\n return `<p>${contentParts.join('<br/>')}</p>`;\n }\n\n private getTextContent(notification: Notification) {\n const contentParts: string[] = [];\n if (notification.payload.description) {\n contentParts.push(notification.payload.description);\n }\n contentParts.push(this.getNotificationLink(notification));\n return contentParts.join('\\n\\n');\n }\n\n private async sendPlainEmail(notification: Notification, emails: string[]) {\n const mailOptions = {\n from: this.sender,\n subject: notification.payload.title,\n html: this.getHtmlContent(notification),\n text: this.getTextContent(notification),\n replyTo: this.replyTo,\n };\n\n await this.sendMails(mailOptions, emails);\n }\n\n private async sendTemplateEmail(\n notification: Notification,\n emails: string[],\n ) {\n const mailOptions = {\n from: this.sender,\n subject:\n (await this.templateRenderer?.getSubject?.(notification)) ??\n notification.payload.title,\n html: await this.templateRenderer?.getHtml?.(notification),\n text: await this.templateRenderer?.getText?.(notification),\n replyTo: this.replyTo,\n };\n\n await this.sendMails(mailOptions, emails);\n }\n\n async postProcess(\n notification: Notification,\n options: NotificationSendOptions,\n ): Promise<void> {\n this.transporter = await this.getTransporter();\n\n let emails: string[] = [];\n try {\n emails = await this.getRecipientEmails(notification, options);\n } catch (e) {\n this.logger.error(`Failed to resolve recipient emails: ${e}`);\n return;\n }\n\n if (emails.length === 0) {\n this.logger.info(\n `No email recipients found for notification: ${notification.id}, skipping`,\n );\n return;\n }\n\n this.logger.debug(`Sending notification emails to: ${emails.join(',')}`);\n\n if (!this.templateRenderer) {\n await this.sendPlainEmail(notification, emails);\n return;\n }\n\n await this.sendTemplateEmail(notification, emails);\n }\n\n getNotificationFilters(): NotificationProcessorFilters {\n return this.filter;\n }\n}\n"],"names":["config","durationToMilliseconds","readDurationFromConfig","getProcessorFiltersFromConfig","createSmtpTransport","DefaultAwsCredentialsManager","createSesTransport","createSendmailTransport","createStreamTransport","createAzureTransport","CATALOG_FILTER_EXISTS","compact","pThrottle"],"mappings":";;;;;;;;;;;;;;;;;;;AAgDO,MAAM,2BAA6D,CAAA;AAAA,EAcxE,YACmB,MACA,EAAAA,QAAA,EACA,OACA,EAAA,IAAA,EACA,OACA,gBACjB,EAAA;AANiB,IAAA,IAAA,CAAA,MAAA,GAAA,MAAA;AACA,IAAA,IAAA,CAAA,MAAA,GAAAA,QAAA;AACA,IAAA,IAAA,CAAA,OAAA,GAAA,OAAA;AACA,IAAA,IAAA,CAAA,IAAA,GAAA,IAAA;AACA,IAAA,IAAA,CAAA,KAAA,GAAA,KAAA;AACA,IAAA,IAAA,CAAA,gBAAA,GAAA,gBAAA;AAEjB,IAAA,MAAM,uBAAuBA,QAAO,CAAA,SAAA;AAAA,MAClC;AAAA,KACF;AACA,IAAK,IAAA,CAAA,eAAA,GAAkB,oBAAqB,CAAA,SAAA,CAAU,iBAAiB,CAAA;AACvE,IAAK,IAAA,CAAA,eAAA,GACH,oBAAqB,CAAA,iBAAA,CAAkB,iBAAiB,CAAA;AAC1D,IAAK,IAAA,CAAA,MAAA,GAAS,oBAAqB,CAAA,SAAA,CAAU,QAAQ,CAAA;AACrD,IAAK,IAAA,CAAA,OAAA,GAAU,oBAAqB,CAAA,iBAAA,CAAkB,SAAS,CAAA;AAC/D,IAAA,IAAA,CAAK,gBACH,GAAA,oBAAA,CAAqB,iBAAkB,CAAA,kBAAkB,CAAK,IAAA,CAAA;AAChE,IAAA,IAAA,CAAK,gBAAmB,GAAA,oBAAA,CAAqB,GAAI,CAAA,kBAAkB,CAC/D,GAAAC,4BAAA;AAAA,MACEC,8BAAuB,oBAAsB,EAAA;AAAA,QAC3C,GAAK,EAAA;AAAA,OACN;AAAA,KAEH,GAAA,GAAA;AACJ,IAAA,IAAA,CAAK,QAAW,GAAA,oBAAA,CAAqB,GAAI,CAAA,WAAW,CAChD,GAAAD,4BAAA;AAAA,MACEC,6BAAuB,CAAA,oBAAA,EAAsB,EAAE,GAAA,EAAK,aAAa;AAAA,KAEnE,GAAA,IAAA;AACJ,IAAK,IAAA,CAAA,eAAA,GAAkBF,QAAO,CAAA,SAAA,CAAU,aAAa,CAAA;AACrD,IAAA,IAAA,CAAK,0BAA0B,oBAAqB,CAAA,sBAAA;AAAA,MAClD;AAAA,KACF;AACA,IAAA,IAAA,CAAK,yBAAyB,oBAAqB,CAAA,sBAAA;AAAA,MACjD;AAAA,KACF;AACA,IAAK,IAAA,CAAA,MAAA,GAASG,wDAA8B,oBAAoB,CAAA;AAAA;AAClE,EAnDQ,WAAA;AAAA,EACS,eAAA;AAAA,EACA,eAAA;AAAA,EACA,MAAA;AAAA,EACA,OAAA;AAAA,EACA,QAAA;AAAA,EACA,gBAAA;AAAA,EACA,gBAAA;AAAA,EACA,eAAA;AAAA,EACA,MAAA;AAAA,EACA,uBAAA;AAAA,EACA,sBAAA;AAAA,EA0CjB,MAAc,cAAiB,GAAA;AAC7B,IAAA,IAAI,KAAK,WAAa,EAAA;AACpB,MAAA,OAAO,IAAK,CAAA,WAAA;AAAA;AAEd,IAAA,MAAM,SAAY,GAAA,IAAA,CAAK,eAAgB,CAAA,SAAA,CAAU,WAAW,CAAA;AAC5D,IAAA,IAAI,cAAc,MAAQ,EAAA;AACxB,MAAK,IAAA,CAAA,WAAA,GAAcC,wBAAoB,CAAA,IAAA,CAAK,eAAe,CAAA;AAAA,KAC7D,MAAA,IAAW,cAAc,KAAO,EAAA;AAC9B,MAAA,MAAM,wBAAwBC,+CAA6B,CAAA,UAAA;AAAA,QACzD,IAAK,CAAA;AAAA,OACP;AACA,MAAA,IAAA,CAAK,cAAc,MAAMC,sBAAA;AAAA,QACvB,IAAK,CAAA,eAAA;AAAA,QACL;AAAA,OACF;AAAA,KACF,MAAA,IAAW,cAAc,UAAY,EAAA;AACnC,MAAK,IAAA,CAAA,WAAA,GAAcC,gCAAwB,CAAA,IAAA,CAAK,eAAe,CAAA;AAAA,KACjE,MAAA,IAAW,cAAc,QAAU,EAAA;AACjC,MAAA,IAAA,CAAK,cAAcC,4BAAsB,EAAA;AAAA,KAC3C,MAAA,IAAW,cAAc,OAAS,EAAA;AAChC,MAAK,IAAA,CAAA,WAAA,GAAcC,0BAAqB,CAAA,IAAA,CAAK,eAAe,CAAA;AAAA,KACvD,MAAA;AACL,MAAA,MAAM,IAAI,KAAA,CAAM,CAA0B,uBAAA,EAAA,SAAS,CAAE,CAAA,CAAA;AAAA;AAEvD,IAAA,OAAO,IAAK,CAAA,WAAA;AAAA;AACd,EAEA,OAAkB,GAAA;AAChB,IAAO,OAAA,OAAA;AAAA;AACT,EAEA,MAAc,kBAAwC,GAAA;AACpD,IAAI,IAAA,CAAC,KAAK,eAAiB,EAAA;AACzB,MAAA,OAAO,EAAC;AAAA;AAGV,IAAA,MAAM,QAAW,GAAA,IAAA,CAAK,eAAgB,CAAA,SAAA,CAAU,UAAU,CAAA;AAC1D,IAAA,IAAI,aAAa,MAAQ,EAAA;AACvB,MAAA,OAAO,EAAC;AAAA;AAGV,IAAA,IAAI,aAAa,QAAU,EAAA;AACzB,MAAA,OACE,IAAK,CAAA,eAAA,CAAgB,sBAAuB,CAAA,gBAAgB,KAAK,EAAC;AAAA;AAItE,IAAA,IAAI,aAAa,OAAS,EAAA;AACxB,MAAA,MAAM,MAAS,GAAA,MAAM,IAAK,CAAA,KAAA,EAAO,IAAc,iBAAiB,CAAA;AAChE,MAAA,IAAI,MAAQ,EAAA;AACV,QAAO,OAAA,MAAA;AAAA;AAGT,MAAM,MAAA,QAAA,GAAW,MAAM,IAAA,CAAK,OAAQ,CAAA,WAAA;AAAA,QAClC;AAAA,UACE,MAAQ,EAAA;AAAA,YACN,EAAE,IAAA,EAAM,MAAQ,EAAA,oBAAA,EAAsBC,mCAAsB;AAAA,WAC9D;AAAA,UACA,MAAA,EAAQ,CAAC,oBAAoB;AAAA,SAC/B;AAAA,QACA,EAAE,WAAa,EAAA,MAAM,IAAK,CAAA,IAAA,CAAK,0BAA2B;AAAA,OAC5D;AACA,MAAA,MAAM,MAAMC,cAAQ,CAAA;AAAA,QAClB,GAAG,IAAI,GAAA;AAAA,UACL,QAAA,CAAS,KAAM,CAAA,GAAA,CAAI,CAAU,MAAA,KAAA;AAC3B,YAAQ,OAAA,MAAA,EAAuB,KAAK,OAAS,EAAA,KAAA;AAAA,WAC9C;AAAA;AACH,OACD,CAAA;AAED,MAAA,MAAM,IAAK,CAAA,KAAA,EAAO,GAAI,CAAA,iBAAA,EAAmB,GAAK,EAAA;AAAA,QAC5C,KAAK,IAAK,CAAA;AAAA,OACX,CAAA;AACD,MAAO,OAAA,GAAA;AAAA;AAGT,IAAA,MAAM,IAAI,KAAA,CAAM,CAAmC,gCAAA,EAAA,QAAQ,CAAE,CAAA,CAAA;AAAA;AAC/D,EAEA,MAAc,aAAa,SAAsC,EAAA;AAC/D,IAAA,MAAM,SAAS,MAAM,IAAA,CAAK,OAAO,GAAc,CAAA,CAAA,YAAA,EAAe,SAAS,CAAE,CAAA,CAAA;AACzE,IAAA,IAAI,MAAQ,EAAA;AACV,MAAO,OAAA,MAAA;AAAA;AAGT,IAAA,MAAM,MAAS,GAAA,MAAM,IAAK,CAAA,OAAA,CAAQ,eAAe,SAAW,EAAA;AAAA,MAC1D,WAAa,EAAA,MAAM,IAAK,CAAA,IAAA,CAAK,wBAAyB;AAAA,KACvD,CAAA;AACD,IAAA,MAAM,MAAgB,EAAC;AACvB,IAAA,IAAI,MAAQ,EAAA;AACV,MAAA,MAAM,UAAa,GAAA,MAAA;AACnB,MAAI,IAAA,UAAA,CAAW,IAAK,CAAA,OAAA,EAAS,KAAO,EAAA;AAClC,QAAA,GAAA,CAAI,IAAK,CAAA,UAAA,CAAW,IAAK,CAAA,OAAA,CAAQ,KAAK,CAAA;AAAA;AACxC;AAGF,IAAA,MAAM,KAAK,KAAO,EAAA,GAAA,CAAI,CAAe,YAAA,EAAA,SAAS,IAAI,GAAK,EAAA;AAAA,MACrD,KAAK,IAAK,CAAA;AAAA,KACX,CAAA;AAED,IAAO,OAAA,GAAA;AAAA;AACT,EAEA,MAAc,kBACZ,CAAA,YAAA,EACA,OACmB,EAAA;AACnB,IAAI,IAAA,MAAA;AACJ,IAAI,IAAA,OAAA,CAAQ,UAAW,CAAA,IAAA,KAAS,WAAa,EAAA;AAC3C,MAAS,MAAA,GAAA,MAAM,KAAK,kBAAmB,EAAA;AAAA,KACzC,MAAA,IAAW,QAAQ,UAAW,CAAA,IAAA,KAAS,YAAY,CAAC,CAAC,aAAa,IAAM,EAAA;AACtE,MAAA,MAAA,GAAS,MAAM,IAAA,CAAK,YAAa,CAAA,YAAA,CAAa,IAAI,CAAA;AAAA,KAC7C,MAAA;AACL,MAAA,IAAA,CAAK,MAAO,CAAA,IAAA;AAAA,QACV,CAAA,0BAAA,EAA6B,OAAQ,CAAA,UAAA,CAAW,IAAI,CAAA,iBAAA;AAAA,OACtD;AACA,MAAA,OAAO,EAAC;AAAA;AAGV,IAAA,IAAI,KAAK,uBAAyB,EAAA;AAChC,MAAA,MAAA,GAAS,MAAO,CAAA,MAAA;AAAA,QAAO,CACrB,KAAA,KAAA,IAAA,CAAK,uBAAyB,EAAA,QAAA,CAAS,KAAK;AAAA,OAC9C;AAAA;AAGF,IAAA,IAAI,KAAK,sBAAwB,EAAA;AAC/B,MAAA,MAAA,GAAS,MAAO,CAAA,MAAA;AAAA,QACd,CAAS,KAAA,KAAA,CAAC,IAAK,CAAA,sBAAA,EAAwB,SAAS,KAAK;AAAA,OACvD;AAAA;AAEF,IAAO,OAAA,MAAA;AAAA;AACT,EAEA,MAAc,SAAS,OAAuB,EAAA;AAC5C,IAAI,IAAA;AACF,MAAA,IAAA,CAAK,MAAO,CAAA,KAAA,CAAM,CAAiC,8BAAA,EAAA,OAAA,CAAQ,EAAE,CAAE,CAAA,CAAA;AAC/D,MAAM,MAAA,IAAA,CAAK,WAAY,CAAA,QAAA,CAAS,OAAO,CAAA;AAAA,aAChC,CAAG,EAAA;AACV,MAAA,IAAA,CAAK,OAAO,KAAM,CAAA,CAAA,wBAAA,EAA2B,QAAQ,EAAE,CAAA,EAAA,EAAK,CAAC,CAAE,CAAA,CAAA;AAAA;AACjE;AACF,EAEA,MAAc,SAAU,CAAA,OAAA,EAAuB,MAAkB,EAAA;AAC/D,IAAA,MAAM,WAAWC,0BAAU,CAAA;AAAA,MACzB,OAAO,IAAK,CAAA,gBAAA;AAAA,MACZ,UAAU,IAAK,CAAA;AAAA,KAChB,CAAA;AAED,IAAA,MAAM,YAAY,QAAS,CAAA,CAAC,SAAuB,IAAK,CAAA,QAAA,CAAS,IAAI,CAAC,CAAA;AACtE,IAAA,MAAM,OAAQ,CAAA,GAAA;AAAA,MACZ,MAAA,CAAO,GAAI,CAAA,CAAA,KAAA,KAAS,SAAU,CAAA,EAAE,GAAG,OAAS,EAAA,EAAA,EAAI,KAAM,EAAC,CAAC;AAAA,KAC1D;AAAA;AACF,EAEQ,oBAAoB,YAA4B,EAAA;AACtD,IAAI,IAAA,YAAA,CAAa,QAAQ,IAAM,EAAA;AAC7B,MAAA,MAAM,oBAAoB,CAAC,CAAA,KAAc,CAAE,CAAA,OAAA,CAAQ,OAAO,EAAE,CAAA;AAC5D,MAAA,MAAM,sBAAsB,CAAC,CAAA,KAAc,CAAE,CAAA,OAAA,CAAQ,QAAQ,GAAG,CAAA;AAEhE,MAAI,IAAA;AACF,QAAA,MAAM,MAAM,IAAI,GAAA;AAAA,UACd,iBAAA,CAAkB,YAAa,CAAA,OAAA,CAAQ,IAAI,CAAA;AAAA,UAC3C,mBAAA,CAAoB,KAAK,eAAe;AAAA,SAC1C;AACA,QAAA,OAAO,IAAI,QAAS,EAAA;AAAA,eACb,EAAI,EAAA;AAAA;AAGb,MAAA,OAAO,aAAa,OAAQ,CAAA,IAAA;AAAA;AAE9B,IAAO,OAAA,CAAA,EAAG,KAAK,eAAe,CAAA,cAAA,CAAA;AAAA;AAChC,EAEQ,eAAe,YAA4B,EAAA;AACjD,IAAA,MAAM,eAAyB,EAAC;AAChC,IAAI,IAAA,YAAA,CAAa,QAAQ,WAAa,EAAA;AACpC,MAAA,YAAA,CAAa,IAAK,CAAA,CAAA,EAAG,YAAa,CAAA,OAAA,CAAQ,WAAW,CAAE,CAAA,CAAA;AAAA;AAEzD,IAAM,MAAA,IAAA,GAAO,IAAK,CAAA,mBAAA,CAAoB,YAAY,CAAA;AAClD,IAAA,YAAA,CAAa,IAAK,CAAA,CAAA,SAAA,EAAY,IAAI,CAAA,EAAA,EAAK,IAAI,CAAM,IAAA,CAAA,CAAA;AACjD,IAAA,OAAO,CAAM,GAAA,EAAA,YAAA,CAAa,IAAK,CAAA,OAAO,CAAC,CAAA,IAAA,CAAA;AAAA;AACzC,EAEQ,eAAe,YAA4B,EAAA;AACjD,IAAA,MAAM,eAAyB,EAAC;AAChC,IAAI,IAAA,YAAA,CAAa,QAAQ,WAAa,EAAA;AACpC,MAAa,YAAA,CAAA,IAAA,CAAK,YAAa,CAAA,OAAA,CAAQ,WAAW,CAAA;AAAA;AAEpD,IAAA,YAAA,CAAa,IAAK,CAAA,IAAA,CAAK,mBAAoB,CAAA,YAAY,CAAC,CAAA;AACxD,IAAO,OAAA,YAAA,CAAa,KAAK,MAAM,CAAA;AAAA;AACjC,EAEA,MAAc,cAAe,CAAA,YAAA,EAA4B,MAAkB,EAAA;AACzE,IAAA,MAAM,WAAc,GAAA;AAAA,MAClB,MAAM,IAAK,CAAA,MAAA;AAAA,MACX,OAAA,EAAS,aAAa,OAAQ,CAAA,KAAA;AAAA,MAC9B,IAAA,EAAM,IAAK,CAAA,cAAA,CAAe,YAAY,CAAA;AAAA,MACtC,IAAA,EAAM,IAAK,CAAA,cAAA,CAAe,YAAY,CAAA;AAAA,MACtC,SAAS,IAAK,CAAA;AAAA,KAChB;AAEA,IAAM,MAAA,IAAA,CAAK,SAAU,CAAA,WAAA,EAAa,MAAM,CAAA;AAAA;AAC1C,EAEA,MAAc,iBACZ,CAAA,YAAA,EACA,MACA,EAAA;AACA,IAAA,MAAM,WAAc,GAAA;AAAA,MAClB,MAAM,IAAK,CAAA,MAAA;AAAA,MACX,OAAA,EACG,MAAM,IAAK,CAAA,gBAAA,EAAkB,aAAa,YAAY,CAAA,IACvD,aAAa,OAAQ,CAAA,KAAA;AAAA,MACvB,IAAM,EAAA,MAAM,IAAK,CAAA,gBAAA,EAAkB,UAAU,YAAY,CAAA;AAAA,MACzD,IAAM,EAAA,MAAM,IAAK,CAAA,gBAAA,EAAkB,UAAU,YAAY,CAAA;AAAA,MACzD,SAAS,IAAK,CAAA;AAAA,KAChB;AAEA,IAAM,MAAA,IAAA,CAAK,SAAU,CAAA,WAAA,EAAa,MAAM,CAAA;AAAA;AAC1C,EAEA,MAAM,WACJ,CAAA,YAAA,EACA,OACe,EAAA;AACf,IAAK,IAAA,CAAA,WAAA,GAAc,MAAM,IAAA,CAAK,cAAe,EAAA;AAE7C,IAAA,IAAI,SAAmB,EAAC;AACxB,IAAI,IAAA;AACF,MAAA,MAAA,GAAS,MAAM,IAAA,CAAK,kBAAmB,CAAA,YAAA,EAAc,OAAO,CAAA;AAAA,aACrD,CAAG,EAAA;AACV,MAAA,IAAA,CAAK,MAAO,CAAA,KAAA,CAAM,CAAuC,oCAAA,EAAA,CAAC,CAAE,CAAA,CAAA;AAC5D,MAAA;AAAA;AAGF,IAAI,IAAA,MAAA,CAAO,WAAW,CAAG,EAAA;AACvB,MAAA,IAAA,CAAK,MAAO,CAAA,IAAA;AAAA,QACV,CAAA,4CAAA,EAA+C,aAAa,EAAE,CAAA,UAAA;AAAA,OAChE;AACA,MAAA;AAAA;AAGF,IAAA,IAAA,CAAK,OAAO,KAAM,CAAA,CAAA,gCAAA,EAAmC,OAAO,IAAK,CAAA,GAAG,CAAC,CAAE,CAAA,CAAA;AAEvE,IAAI,IAAA,CAAC,KAAK,gBAAkB,EAAA;AAC1B,MAAM,MAAA,IAAA,CAAK,cAAe,CAAA,YAAA,EAAc,MAAM,CAAA;AAC9C,MAAA;AAAA;AAGF,IAAM,MAAA,IAAA,CAAK,iBAAkB,CAAA,YAAA,EAAc,MAAM,CAAA;AAAA;AACnD,EAEA,sBAAuD,GAAA;AACrD,IAAA,OAAO,IAAK,CAAA,MAAA;AAAA;AAEhB;;;;"}
|
|
1
|
+
{"version":3,"file":"NotificationsEmailProcessor.cjs.js","sources":["../../src/processor/NotificationsEmailProcessor.ts"],"sourcesContent":["/*\n * Copyright 2024 The Backstage Authors\n *\n * Licensed under the Apache License, Version 2.0 (the \"License\");\n * you may not use this file except in compliance with the License.\n * You may obtain a copy of the License at\n *\n * http://www.apache.org/licenses/LICENSE-2.0\n *\n * Unless required by applicable law or agreed to in writing, software\n * distributed under the License is distributed on an \"AS IS\" BASIS,\n * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.\n * See the License for the specific language governing permissions and\n * limitations under the License.\n */\n\nimport {\n NotificationProcessor,\n NotificationSendOptions,\n} from '@backstage/plugin-notifications-node';\nimport {\n AuthService,\n CacheService,\n LoggerService,\n} from '@backstage/backend-plugin-api';\nimport { Config, readDurationFromConfig } from '@backstage/config';\nimport { durationToMilliseconds } from '@backstage/types';\nimport { CATALOG_FILTER_EXISTS } from '@backstage/catalog-client';\nimport {\n getProcessorFiltersFromConfig,\n Notification,\n NotificationProcessorFilters,\n} from '@backstage/plugin-notifications-common';\nimport {\n createAzureTransport,\n createSendmailTransport,\n createSesTransport,\n createSmtpTransport,\n createStreamTransport,\n} from './transports';\nimport { UserEntity } from '@backstage/catalog-model';\nimport { CatalogService } from '@backstage/plugin-catalog-node';\nimport { compact } from 'lodash';\nimport { DefaultAwsCredentialsManager } from '@backstage/integration-aws-node';\nimport { NotificationTemplateRenderer } from '../extensions';\nimport Mail from 'nodemailer/lib/mailer';\nimport pThrottle from 'p-throttle';\n\nexport class NotificationsEmailProcessor implements NotificationProcessor {\n private transporter: any;\n private readonly broadcastConfig?: Config;\n private readonly transportConfig: Config;\n private readonly sender: string;\n private readonly replyTo?: string;\n private readonly sesConfig?: Config;\n private readonly cacheTtl: number;\n private readonly concurrencyLimit: number;\n private readonly throttleInterval: number;\n private readonly frontendBaseUrl: string;\n private readonly filter: NotificationProcessorFilters;\n private readonly allowlistEmailAddresses?: string[];\n private readonly denylistEmailAddresses?: string[];\n\n constructor(\n private readonly logger: LoggerService,\n private readonly config: Config,\n private readonly catalog: CatalogService,\n private readonly auth: AuthService,\n private readonly cache?: CacheService,\n private readonly templateRenderer?: NotificationTemplateRenderer,\n ) {\n const emailProcessorConfig = config.getConfig(\n 'notifications.processors.email',\n );\n this.transportConfig = emailProcessorConfig.getConfig('transportConfig');\n this.broadcastConfig =\n emailProcessorConfig.getOptionalConfig('broadcastConfig');\n this.sender = emailProcessorConfig.getString('sender');\n this.replyTo = emailProcessorConfig.getOptionalString('replyTo');\n this.sesConfig = emailProcessorConfig.getOptionalConfig('sesConfig');\n this.concurrencyLimit =\n emailProcessorConfig.getOptionalNumber('concurrencyLimit') ?? 2;\n this.throttleInterval = emailProcessorConfig.has('throttleInterval')\n ? durationToMilliseconds(\n readDurationFromConfig(emailProcessorConfig, {\n key: 'throttleInterval',\n }),\n )\n : 100;\n this.cacheTtl = emailProcessorConfig.has('cache.ttl')\n ? durationToMilliseconds(\n readDurationFromConfig(emailProcessorConfig, { key: 'cache.ttl' }),\n )\n : 3_600_000;\n this.frontendBaseUrl = config.getString('app.baseUrl');\n this.allowlistEmailAddresses = emailProcessorConfig.getOptionalStringArray(\n 'allowlistEmailAddresses',\n );\n this.denylistEmailAddresses = emailProcessorConfig.getOptionalStringArray(\n 'denylistEmailAddresses',\n );\n this.filter = getProcessorFiltersFromConfig(emailProcessorConfig);\n }\n\n private async getTransporter() {\n if (this.transporter) {\n return this.transporter;\n }\n const transport = this.transportConfig.getString('transport');\n if (transport === 'smtp') {\n this.transporter = createSmtpTransport(this.transportConfig);\n } else if (transport === 'ses') {\n const awsCredentialsManager = DefaultAwsCredentialsManager.fromConfig(\n this.config,\n );\n this.transporter = await createSesTransport(\n this.transportConfig,\n awsCredentialsManager,\n );\n } else if (transport === 'sendmail') {\n this.transporter = createSendmailTransport(this.transportConfig);\n } else if (transport === 'stream') {\n this.transporter = createStreamTransport();\n } else if (transport === 'azure') {\n this.transporter = createAzureTransport(this.transportConfig);\n } else {\n throw new Error(`Unsupported transport: ${transport}`);\n }\n return this.transporter;\n }\n\n getName(): string {\n return 'Email';\n }\n\n private async getBroadcastEmails(): Promise<string[]> {\n if (!this.broadcastConfig) {\n return [];\n }\n\n const receiver = this.broadcastConfig.getString('receiver');\n if (receiver === 'none') {\n return [];\n }\n\n if (receiver === 'config') {\n return (\n this.broadcastConfig.getOptionalStringArray('receiverEmails') ?? []\n );\n }\n\n if (receiver === 'users') {\n const cached = await this.cache?.get<string[]>('user-emails:all');\n if (cached) {\n return cached;\n }\n\n const entities = await this.catalog.getEntities(\n {\n filter: [\n { kind: 'user', 'spec.profile.email': CATALOG_FILTER_EXISTS },\n ],\n fields: ['spec.profile.email'],\n },\n { credentials: await this.auth.getOwnServiceCredentials() },\n );\n const ret = compact([\n ...new Set(\n entities.items.map(entity => {\n return (entity as UserEntity)?.spec.profile?.email;\n }),\n ),\n ]);\n\n await this.cache?.set('user-emails:all', ret, {\n ttl: this.cacheTtl,\n });\n return ret;\n }\n\n throw new Error(`Unsupported broadcast receiver: ${receiver}`);\n }\n\n private async getUserEmail(entityRef: string): Promise<string[]> {\n const cached = await this.cache?.get<string[]>(`user-emails:${entityRef}`);\n if (cached) {\n return cached;\n }\n\n const entity = await this.catalog.getEntityByRef(entityRef, {\n credentials: await this.auth.getOwnServiceCredentials(),\n });\n const ret: string[] = [];\n if (entity) {\n const userEntity = entity as UserEntity;\n if (userEntity.spec.profile?.email) {\n ret.push(userEntity.spec.profile.email);\n }\n }\n\n await this.cache?.set(`user-emails:${entityRef}`, ret, {\n ttl: this.cacheTtl,\n });\n\n return ret;\n }\n\n private async getRecipientEmails(\n notification: Notification,\n options: NotificationSendOptions,\n ): Promise<string[]> {\n let emails: string[];\n if (options.recipients.type === 'broadcast') {\n emails = await this.getBroadcastEmails();\n } else if (options.recipients.type === 'entity' && !!notification.user) {\n emails = await this.getUserEmail(notification.user);\n } else {\n this.logger.info(\n `Unknown notification type ${options.recipients.type} or missing user.`,\n );\n return [];\n }\n\n if (this.allowlistEmailAddresses) {\n emails = emails.filter(email =>\n this.allowlistEmailAddresses?.includes(email),\n );\n }\n\n if (this.denylistEmailAddresses) {\n emails = emails.filter(\n email => !this.denylistEmailAddresses?.includes(email),\n );\n }\n return emails;\n }\n\n private async sendMail(options: Mail.Options) {\n try {\n this.logger.debug(`Sending notification email to ${options.to}`);\n await this.transporter.sendMail(options);\n } catch (e) {\n this.logger.error(`Failed to send email to ${options.to}: ${e}`);\n }\n }\n\n private async sendMails(options: Mail.Options, emails: string[]) {\n const throttle = pThrottle({\n limit: this.concurrencyLimit,\n interval: this.throttleInterval,\n });\n\n const throttled = throttle((opts: Mail.Options) => this.sendMail(opts));\n await Promise.all(\n emails.map(email => throttled({ ...options, to: email })),\n );\n }\n\n private getNotificationLink(notification: Notification) {\n if (notification.payload.link) {\n const stripLeadingSlash = (s: string) => s.replace(/^\\//, '');\n const ensureTrailingSlash = (s: string) => s.replace(/\\/?$/, '/');\n\n try {\n const url = new URL(\n stripLeadingSlash(notification.payload.link),\n ensureTrailingSlash(this.frontendBaseUrl),\n );\n return url.toString();\n } catch (_e) {\n // noop: fallback to relative URL\n }\n return notification.payload.link;\n }\n return `${this.frontendBaseUrl}/notifications`;\n }\n\n private getHtmlContent(notification: Notification) {\n const contentParts: string[] = [];\n if (notification.payload.description) {\n contentParts.push(`${notification.payload.description}`);\n }\n const link = this.getNotificationLink(notification);\n contentParts.push(`<a href=\"${link}\">${link}</a>`);\n return `<p>${contentParts.join('<br/>')}</p>`;\n }\n\n private getTextContent(notification: Notification) {\n const contentParts: string[] = [];\n if (notification.payload.description) {\n contentParts.push(notification.payload.description);\n }\n contentParts.push(this.getNotificationLink(notification));\n return contentParts.join('\\n\\n');\n }\n\n private async getSesOptions() {\n if (!this.sesConfig) {\n return undefined;\n }\n const ses: Record<string, string> = {};\n const sourceArn = this.sesConfig.getOptionalString('sourceArn');\n const fromArn = this.sesConfig.getOptionalString('fromArn');\n const configurationSetName = this.sesConfig.getOptionalString(\n 'configurationSetName',\n );\n\n if (sourceArn) ses.SourceArn = sourceArn;\n if (fromArn) ses.FromArn = fromArn;\n if (configurationSetName) ses.ConfigurationSetName = configurationSetName;\n\n return Object.keys(ses).length > 0 ? ses : undefined;\n }\n\n private async sendPlainEmail(notification: Notification, emails: string[]) {\n const mailOptions = {\n from: this.sender,\n subject: notification.payload.title,\n html: this.getHtmlContent(notification),\n text: this.getTextContent(notification),\n replyTo: this.replyTo,\n ses: await this.getSesOptions(),\n };\n\n await this.sendMails(mailOptions, emails);\n }\n\n private async sendTemplateEmail(\n notification: Notification,\n emails: string[],\n ) {\n const mailOptions = {\n from: this.sender,\n subject:\n (await this.templateRenderer?.getSubject?.(notification)) ??\n notification.payload.title,\n html: await this.templateRenderer?.getHtml?.(notification),\n text: await this.templateRenderer?.getText?.(notification),\n replyTo: this.replyTo,\n ses: await this.getSesOptions(),\n };\n\n await this.sendMails(mailOptions, emails);\n }\n\n async postProcess(\n notification: Notification,\n options: NotificationSendOptions,\n ): Promise<void> {\n this.transporter = await this.getTransporter();\n\n let emails: string[] = [];\n try {\n emails = await this.getRecipientEmails(notification, options);\n } catch (e) {\n this.logger.error(`Failed to resolve recipient emails: ${e}`);\n return;\n }\n\n if (emails.length === 0) {\n this.logger.info(\n `No email recipients found for notification: ${notification.id}, skipping`,\n );\n return;\n }\n\n this.logger.debug(`Sending notification emails to: ${emails.join(',')}`);\n\n if (!this.templateRenderer) {\n await this.sendPlainEmail(notification, emails);\n return;\n }\n\n await this.sendTemplateEmail(notification, emails);\n }\n\n getNotificationFilters(): NotificationProcessorFilters {\n return this.filter;\n }\n}\n"],"names":["config","durationToMilliseconds","readDurationFromConfig","getProcessorFiltersFromConfig","createSmtpTransport","DefaultAwsCredentialsManager","createSesTransport","createSendmailTransport","createStreamTransport","createAzureTransport","CATALOG_FILTER_EXISTS","compact","pThrottle"],"mappings":"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"}
|
package/package.json
CHANGED
|
@@ -1,6 +1,6 @@
|
|
|
1
1
|
{
|
|
2
2
|
"name": "@backstage/plugin-notifications-backend-module-email",
|
|
3
|
-
"version": "0.3.
|
|
3
|
+
"version": "0.3.11-next.1",
|
|
4
4
|
"description": "The email backend module for the notifications plugin.",
|
|
5
5
|
"backstage": {
|
|
6
6
|
"role": "backend-plugin-module",
|
|
@@ -41,22 +41,22 @@
|
|
|
41
41
|
"@aws-sdk/types": "^3.347.0",
|
|
42
42
|
"@azure/communication-email": "^1.0.0",
|
|
43
43
|
"@azure/identity": "^4.0.0",
|
|
44
|
-
"@backstage/backend-plugin-api": "
|
|
45
|
-
"@backstage/catalog-client": "
|
|
46
|
-
"@backstage/catalog-model": "
|
|
47
|
-
"@backstage/config": "
|
|
48
|
-
"@backstage/integration-aws-node": "
|
|
49
|
-
"@backstage/plugin-catalog-node": "
|
|
50
|
-
"@backstage/plugin-notifications-common": "
|
|
51
|
-
"@backstage/plugin-notifications-node": "
|
|
52
|
-
"@backstage/types": "
|
|
44
|
+
"@backstage/backend-plugin-api": "1.4.1-next.0",
|
|
45
|
+
"@backstage/catalog-client": "1.10.2-next.0",
|
|
46
|
+
"@backstage/catalog-model": "1.7.5-next.0",
|
|
47
|
+
"@backstage/config": "1.3.3-next.0",
|
|
48
|
+
"@backstage/integration-aws-node": "0.1.17-next.0",
|
|
49
|
+
"@backstage/plugin-catalog-node": "1.17.2-next.0",
|
|
50
|
+
"@backstage/plugin-notifications-common": "0.0.10-next.0",
|
|
51
|
+
"@backstage/plugin-notifications-node": "0.2.17-next.0",
|
|
52
|
+
"@backstage/types": "1.2.1",
|
|
53
53
|
"lodash": "^4.17.21",
|
|
54
54
|
"nodemailer": "^6.9.13",
|
|
55
55
|
"p-throttle": "^4.1.1"
|
|
56
56
|
},
|
|
57
57
|
"devDependencies": {
|
|
58
|
-
"@backstage/backend-test-utils": "
|
|
59
|
-
"@backstage/cli": "
|
|
58
|
+
"@backstage/backend-test-utils": "1.7.0-next.1",
|
|
59
|
+
"@backstage/cli": "0.33.1-next.2",
|
|
60
60
|
"@types/nodemailer": "^6.4.14"
|
|
61
61
|
},
|
|
62
62
|
"configSchema": "config.d.ts",
|