diff options
Diffstat (limited to 'deno/mail-relay/aws/deliver.ts')
-rw-r--r-- | deno/mail-relay/aws/deliver.ts | 114 |
1 files changed, 114 insertions, 0 deletions
diff --git a/deno/mail-relay/aws/deliver.ts b/deno/mail-relay/aws/deliver.ts new file mode 100644 index 0000000..0db5fa8 --- /dev/null +++ b/deno/mail-relay/aws/deliver.ts @@ -0,0 +1,114 @@ +// spellchecker: words sesv2 amazonses + +import { SendEmailCommand, SESv2Client } from "@aws-sdk/client-sesv2"; + +import log from "../log.ts"; +import { DbService } from "../db.ts"; +import { + Mail, + MailDeliverContext, + MailDeliverHook, + SyncMailDeliverer, +} from "../mail.ts"; +import { AwsContext } from "./context.ts"; + +declare module "../mail.ts" { + interface MailDeliverResult { + awsMessageId?: string; + } +} + +export class AwsMailMessageIdRewriteHook implements MailDeliverHook { + readonly #db; + + constructor(db: DbService) { + this.#db = db; + } + + async callback(context: MailDeliverContext): Promise<void> { + log.info("Rewrite message ids..."); + const addresses = context.mail.simpleFindAllAddresses(); + log.info(`Addresses found in mail: ${addresses.join(", ")}.`); + for (const address of addresses) { + const awsMessageId = await this.#db.messageIdToAws(address); + if (awsMessageId != null && awsMessageId.length !== 0) { + log.info(`Rewrite ${address} to ${awsMessageId}.`); + context.mail.raw = context.mail.raw.replaceAll(address, awsMessageId); + } + } + log.info("Done rewrite message ids."); + } +} + +export class AwsMailMessageIdSaveHook implements MailDeliverHook { + readonly #db; + + constructor(db: DbService) { + this.#db = db; + } + + async callback(context: MailDeliverContext): Promise<void> { + log.info("Save aws message ids..."); + const messageId = context.mail.startSimpleParse().sections().headers() + .messageId(); + if (messageId == null) { + log.info("Original mail does not have message id. Skip saving."); + return; + } + if (context.result.awsMessageId != null) { + log.info(`Saving ${messageId} => ${context.result.awsMessageId}.`); + await this.#db.addMessageIdMap({ + message_id: messageId, + aws_message_id: context.result.awsMessageId, + }); + } + log.info("Done save message ids."); + } +} + +export class AwsMailDeliverer extends SyncMailDeliverer { + readonly name = "aws"; + readonly #aws; + readonly #ses; + + constructor(aws: AwsContext) { + super(); + this.#aws = aws; + this.#ses = new SESv2Client(aws); + } + + protected override async doDeliver( + mail: Mail, + context: MailDeliverContext, + ): Promise<void> { + log.info("Begin to call aws send-email api..."); + + try { + const sendCommand = new SendEmailCommand({ + Content: { + Raw: { Data: mail.toUtf8Bytes() }, + }, + }); + + const res = await this.#ses.send(sendCommand); + if (res.MessageId == null) { + log.warn("Aws send-email returns no message id."); + } else { + context.result.awsMessageId = + `${res.MessageId}@${this.#aws.region}.amazonses.com`; + } + + context.result.recipients.set("*", { + kind: "done", + message: + `Successfully called aws send-email, message id ${context.result.awsMessageId}.`, + }); + } catch (cause) { + context.result.recipients.set("*", { + kind: "fail", + message: "An error was thrown when calling aws send-email." + cause, + cause, + }); + } + } +} |