Add ses hook parser to capture all the events
This commit is contained in:
189
apps/web/src/server/aws/ses.ts
Normal file
189
apps/web/src/server/aws/ses.ts
Normal file
@@ -0,0 +1,189 @@
|
||||
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: `mail.${domain}`,
|
||||
});
|
||||
|
||||
const emailIdentityResponse = await sesClient.send(emailIdentityCommand);
|
||||
|
||||
if (
|
||||
response.$metadata.httpStatusCode !== 200 ||
|
||||
emailIdentityResponse.$metadata.httpStatusCode !== 200
|
||||
) {
|
||||
console.log(response);
|
||||
console.log(emailIdentityResponse);
|
||||
throw new Error("Failed to create domain identity");
|
||||
}
|
||||
|
||||
return publicKey;
|
||||
}
|
||||
|
||||
export async function deleteDomain(domain: string, region = "us-east-1") {
|
||||
const sesClient = getSesClient(region);
|
||||
const command = new DeleteEmailIdentityCommand({
|
||||
EmailIdentity: domain,
|
||||
});
|
||||
const response = await sesClient.send(command);
|
||||
return response.$metadata.httpStatusCode === 200;
|
||||
}
|
||||
|
||||
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: {
|
||||
Enabled: true,
|
||||
MatchingEventTypes: eventTypes,
|
||||
SnsDestination: {
|
||||
TopicArn: topicArn,
|
||||
},
|
||||
},
|
||||
});
|
||||
|
||||
const response = await sesClient.send(command);
|
||||
return response.$metadata.httpStatusCode === 200;
|
||||
}
|
Reference in New Issue
Block a user