180 lines
4.7 KiB
TypeScript
180 lines
4.7 KiB
TypeScript
import {
|
|
SESv2Client,
|
|
CreateEmailIdentityCommand,
|
|
DeleteEmailIdentityCommand,
|
|
GetEmailIdentityCommand,
|
|
PutEmailIdentityMailFromAttributesCommand,
|
|
SendEmailCommand,
|
|
CreateConfigurationSetEventDestinationCommand,
|
|
CreateConfigurationSetCommand,
|
|
EventType,
|
|
} from "@aws-sdk/client-sesv2";
|
|
import { generateKeyPairSync } from "crypto";
|
|
import { env } from "~/env";
|
|
import { EmailContent } from "~/types";
|
|
import { APP_SETTINGS } from "~/utils/constants";
|
|
|
|
function getSesClient(region = "us-east-1") {
|
|
return new SESv2Client({
|
|
region: region,
|
|
credentials: {
|
|
accessKeyId: env.AWS_ACCESS_KEY,
|
|
secretAccessKey: env.AWS_SECRET_KEY,
|
|
},
|
|
});
|
|
}
|
|
|
|
function generateKeyPair() {
|
|
const { privateKey, publicKey } = generateKeyPairSync("rsa", {
|
|
modulusLength: 2048, // Length of your key in bits
|
|
publicKeyEncoding: {
|
|
type: "spki", // Recommended to be 'spki' by the Node.js docs
|
|
format: "pem",
|
|
},
|
|
privateKeyEncoding: {
|
|
type: "pkcs8", // Recommended to be 'pkcs8' by the Node.js docs
|
|
format: "pem",
|
|
},
|
|
});
|
|
|
|
const base64PrivateKey = privateKey
|
|
.replace("-----BEGIN PRIVATE KEY-----", "")
|
|
.replace("-----END PRIVATE KEY-----", "")
|
|
.replace(/\n/g, "");
|
|
|
|
const base64PublicKey = publicKey
|
|
.replace("-----BEGIN PUBLIC KEY-----", "")
|
|
.replace("-----END PUBLIC KEY-----", "")
|
|
.replace(/\n/g, "");
|
|
|
|
return { privateKey: base64PrivateKey, publicKey: base64PublicKey };
|
|
}
|
|
|
|
export async function addDomain(domain: string, region = "us-east-1") {
|
|
const sesClient = getSesClient(region);
|
|
|
|
const { privateKey, publicKey } = generateKeyPair();
|
|
const command = new CreateEmailIdentityCommand({
|
|
EmailIdentity: domain,
|
|
DkimSigningAttributes: {
|
|
DomainSigningSelector: "unsend",
|
|
DomainSigningPrivateKey: privateKey,
|
|
},
|
|
ConfigurationSetName: APP_SETTINGS.SES_CONFIGURATION_GENERAL,
|
|
});
|
|
const response = await sesClient.send(command);
|
|
|
|
const emailIdentityCommand = new PutEmailIdentityMailFromAttributesCommand({
|
|
EmailIdentity: domain,
|
|
MailFromDomain: `send.${domain}`,
|
|
});
|
|
|
|
const emailIdentityResponse = await sesClient.send(emailIdentityCommand);
|
|
|
|
if (
|
|
response.$metadata.httpStatusCode !== 200 ||
|
|
emailIdentityResponse.$metadata.httpStatusCode !== 200
|
|
) {
|
|
throw new Error("Failed to create email identity");
|
|
}
|
|
|
|
return publicKey;
|
|
}
|
|
|
|
export async function getDomainIdentity(domain: string, region = "us-east-1") {
|
|
const sesClient = getSesClient(region);
|
|
const command = new GetEmailIdentityCommand({
|
|
EmailIdentity: domain,
|
|
});
|
|
const response = await sesClient.send(command);
|
|
return response;
|
|
}
|
|
|
|
export async function sendEmailThroughSes({
|
|
to,
|
|
from,
|
|
subject,
|
|
text,
|
|
html,
|
|
region = "us-east-1",
|
|
configurationSetName,
|
|
}: EmailContent & {
|
|
region?: string;
|
|
configurationSetName: string;
|
|
}) {
|
|
const sesClient = getSesClient(region);
|
|
const command = new SendEmailCommand({
|
|
FromEmailAddress: from,
|
|
Destination: {
|
|
ToAddresses: [to],
|
|
},
|
|
Content: {
|
|
// EmailContent
|
|
Simple: {
|
|
// Message
|
|
Subject: {
|
|
// Content
|
|
Data: subject, // required
|
|
Charset: "UTF-8",
|
|
},
|
|
Body: {
|
|
// Body
|
|
Text: {
|
|
Data: text, // required
|
|
Charset: "UTF-8",
|
|
},
|
|
Html: {
|
|
Data: html, // required
|
|
Charset: "UTF-8",
|
|
},
|
|
},
|
|
},
|
|
},
|
|
ConfigurationSetName: configurationSetName,
|
|
});
|
|
|
|
try {
|
|
const response = await sesClient.send(command);
|
|
console.log("Email sent! Message ID:", response.MessageId);
|
|
return response.MessageId;
|
|
} catch (error) {
|
|
console.error("Failed to send email", error);
|
|
throw new Error("Failed to send email");
|
|
}
|
|
}
|
|
|
|
export async function addWebhookConfiguration(
|
|
configName: string,
|
|
topicArn: string,
|
|
eventTypes: EventType[],
|
|
region = "us-east-1"
|
|
) {
|
|
const sesClient = getSesClient(region);
|
|
|
|
const configSetCommand = new CreateConfigurationSetCommand({
|
|
ConfigurationSetName: configName,
|
|
});
|
|
|
|
const configSetResponse = await sesClient.send(configSetCommand);
|
|
|
|
if (configSetResponse.$metadata.httpStatusCode !== 200) {
|
|
throw new Error("Failed to create configuration set");
|
|
}
|
|
|
|
const command = new CreateConfigurationSetEventDestinationCommand({
|
|
ConfigurationSetName: configName, // required
|
|
EventDestinationName: "unsend_destination", // required
|
|
EventDestination: {
|
|
// EventDestinationDefinition
|
|
Enabled: true,
|
|
MatchingEventTypes: eventTypes,
|
|
SnsDestination: {
|
|
TopicArn: topicArn,
|
|
},
|
|
},
|
|
});
|
|
|
|
const response = await sesClient.send(command);
|
|
return response.$metadata.httpStatusCode === 200;
|
|
}
|