mirror of
https://github.com/documenso/documenso.git
synced 2025-11-13 00:03:33 +10:00
fix: dogfood resend transport
This commit is contained in:
12
package-lock.json
generated
12
package-lock.json
generated
@ -1869,6 +1869,17 @@
|
|||||||
"resolved": "apps/marketing",
|
"resolved": "apps/marketing",
|
||||||
"link": true
|
"link": true
|
||||||
},
|
},
|
||||||
|
"node_modules/@documenso/nodemailer-resend": {
|
||||||
|
"version": "1.0.0",
|
||||||
|
"resolved": "https://registry.npmjs.org/@documenso/nodemailer-resend/-/nodemailer-resend-1.0.0.tgz",
|
||||||
|
"integrity": "sha512-rG+jBbBEsVJUBU6v/2hb+OQD1m3Lhn49TOzQjln73zzL1B/sZsHhYOKpNPlTX0/FafCP7P9fKerndEeIKn54Vw==",
|
||||||
|
"dependencies": {
|
||||||
|
"resend": "^1.1.0"
|
||||||
|
},
|
||||||
|
"peerDependencies": {
|
||||||
|
"nodemailer": "^6.9.3"
|
||||||
|
}
|
||||||
|
},
|
||||||
"node_modules/@documenso/prettier-config": {
|
"node_modules/@documenso/prettier-config": {
|
||||||
"resolved": "packages/prettier-config",
|
"resolved": "packages/prettier-config",
|
||||||
"link": true
|
"link": true
|
||||||
@ -20014,6 +20025,7 @@
|
|||||||
"version": "1.0.0",
|
"version": "1.0.0",
|
||||||
"license": "MIT",
|
"license": "MIT",
|
||||||
"dependencies": {
|
"dependencies": {
|
||||||
|
"@documenso/nodemailer-resend": "1.0.0",
|
||||||
"@react-email/components": "^0.0.7",
|
"@react-email/components": "^0.0.7",
|
||||||
"nodemailer": "^6.9.3",
|
"nodemailer": "^6.9.3",
|
||||||
"react-email": "^1.9.4",
|
"react-email": "^1.9.4",
|
||||||
|
|||||||
@ -1,7 +1,8 @@
|
|||||||
import { createTransport } from 'nodemailer';
|
import { createTransport } from 'nodemailer';
|
||||||
|
|
||||||
|
import { ResendTransport } from '@documenso/nodemailer-resend';
|
||||||
|
|
||||||
import { MailChannelsTransport } from './transports/mailchannels';
|
import { MailChannelsTransport } from './transports/mailchannels';
|
||||||
import { ResendTransport } from './transports/resend';
|
|
||||||
|
|
||||||
const getTransport = () => {
|
const getTransport = () => {
|
||||||
const transport = process.env.NEXT_PRIVATE_SMTP_TRANSPORT ?? 'smtp-auth';
|
const transport = process.env.NEXT_PRIVATE_SMTP_TRANSPORT ?? 'smtp-auth';
|
||||||
|
|||||||
@ -17,6 +17,7 @@
|
|||||||
"worker:test": "tsup worker/index.ts --format esm"
|
"worker:test": "tsup worker/index.ts --format esm"
|
||||||
},
|
},
|
||||||
"dependencies": {
|
"dependencies": {
|
||||||
|
"@documenso/nodemailer-resend": "1.0.0",
|
||||||
"@react-email/components": "^0.0.7",
|
"@react-email/components": "^0.0.7",
|
||||||
"nodemailer": "^6.9.3",
|
"nodemailer": "^6.9.3",
|
||||||
"react-email": "^1.9.4",
|
"react-email": "^1.9.4",
|
||||||
|
|||||||
@ -1,145 +0,0 @@
|
|||||||
import { type SentMessageInfo, type Transport } from 'nodemailer';
|
|
||||||
import type Mail from 'nodemailer/lib/mailer';
|
|
||||||
import type MailMessage from 'nodemailer/lib/mailer/mail-message';
|
|
||||||
import { Resend } from 'resend';
|
|
||||||
|
|
||||||
const VERSION = '1.0.0';
|
|
||||||
|
|
||||||
type ResendTransportOptions = {
|
|
||||||
apiKey: string;
|
|
||||||
};
|
|
||||||
|
|
||||||
type ResendResponseError = {
|
|
||||||
statusCode: number;
|
|
||||||
name: string;
|
|
||||||
message: string;
|
|
||||||
};
|
|
||||||
|
|
||||||
const isResendResponseError = (error: unknown): error is ResendResponseError => {
|
|
||||||
// We could use Zod here, but it's not worth the extra bundle size
|
|
||||||
return (
|
|
||||||
typeof error === 'object' &&
|
|
||||||
error !== null &&
|
|
||||||
'statusCode' in error &&
|
|
||||||
typeof error.statusCode === 'number' &&
|
|
||||||
'name' in error &&
|
|
||||||
typeof error.name === 'string' &&
|
|
||||||
'message' in error &&
|
|
||||||
typeof error.message === 'string'
|
|
||||||
);
|
|
||||||
};
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Transport for sending email via the Resend SDK.
|
|
||||||
*/
|
|
||||||
export class ResendTransport implements Transport<SentMessageInfo> {
|
|
||||||
public name = 'ResendMailTransport';
|
|
||||||
public version = VERSION;
|
|
||||||
|
|
||||||
private _client: Resend;
|
|
||||||
private _options: ResendTransportOptions;
|
|
||||||
|
|
||||||
public static makeTransport(options: Partial<ResendTransportOptions>) {
|
|
||||||
return new ResendTransport(options);
|
|
||||||
}
|
|
||||||
|
|
||||||
constructor(options: Partial<ResendTransportOptions>) {
|
|
||||||
const { apiKey = '' } = options;
|
|
||||||
|
|
||||||
this._options = {
|
|
||||||
apiKey,
|
|
||||||
};
|
|
||||||
|
|
||||||
this._client = new Resend(apiKey);
|
|
||||||
}
|
|
||||||
|
|
||||||
public send(mail: MailMessage, callback: (_err: Error | null, _info: SentMessageInfo) => void) {
|
|
||||||
if (!mail.data.to || !mail.data.from) {
|
|
||||||
return callback(new Error('Missing required fields "to" or "from"'), null);
|
|
||||||
}
|
|
||||||
|
|
||||||
this._client
|
|
||||||
.sendEmail({
|
|
||||||
subject: mail.data.subject ?? '',
|
|
||||||
from: this.toResendFromAddress(mail.data.from),
|
|
||||||
to: this.toResendAddresses(mail.data.to),
|
|
||||||
cc: this.toResendAddresses(mail.data.cc),
|
|
||||||
bcc: this.toResendAddresses(mail.data.bcc),
|
|
||||||
html: mail.data.html?.toString() || '',
|
|
||||||
text: mail.data.text?.toString() || '',
|
|
||||||
attachments: this.toResendAttachments(mail.data.attachments),
|
|
||||||
})
|
|
||||||
.then((response) => {
|
|
||||||
if (isResendResponseError(response)) {
|
|
||||||
throw new Error(`[${response.statusCode}]: ${response.name} ${response.message}`);
|
|
||||||
}
|
|
||||||
|
|
||||||
callback(null, response);
|
|
||||||
})
|
|
||||||
.catch((error) => {
|
|
||||||
callback(error, null);
|
|
||||||
});
|
|
||||||
}
|
|
||||||
|
|
||||||
private toResendAddresses(addresses: Mail.Options['to']) {
|
|
||||||
if (!addresses) {
|
|
||||||
return [];
|
|
||||||
}
|
|
||||||
|
|
||||||
if (typeof addresses === 'string') {
|
|
||||||
return [addresses];
|
|
||||||
}
|
|
||||||
|
|
||||||
if (Array.isArray(addresses)) {
|
|
||||||
return addresses.map((address) => {
|
|
||||||
if (typeof address === 'string') {
|
|
||||||
return address;
|
|
||||||
}
|
|
||||||
|
|
||||||
return address.address;
|
|
||||||
});
|
|
||||||
}
|
|
||||||
|
|
||||||
return [addresses.address];
|
|
||||||
}
|
|
||||||
|
|
||||||
private toResendFromAddress(address: Mail.Options['from']) {
|
|
||||||
if (!address) {
|
|
||||||
return '';
|
|
||||||
}
|
|
||||||
|
|
||||||
if (typeof address === 'string') {
|
|
||||||
return address;
|
|
||||||
}
|
|
||||||
|
|
||||||
return `${address.name} <${address.address}>`;
|
|
||||||
}
|
|
||||||
|
|
||||||
private toResendAttachments(attachments: Mail.Options['attachments']) {
|
|
||||||
if (!attachments) {
|
|
||||||
return [];
|
|
||||||
}
|
|
||||||
|
|
||||||
return attachments.map((attachment) => {
|
|
||||||
if (!attachment.filename || !attachment.content) {
|
|
||||||
throw new Error('Attachment is missing filename or content');
|
|
||||||
}
|
|
||||||
|
|
||||||
if (typeof attachment.content === 'string') {
|
|
||||||
return {
|
|
||||||
filename: attachment.filename,
|
|
||||||
content: Buffer.from(attachment.content),
|
|
||||||
};
|
|
||||||
}
|
|
||||||
|
|
||||||
if (attachment.content instanceof Buffer) {
|
|
||||||
return {
|
|
||||||
filename: attachment.filename,
|
|
||||||
content: attachment.content,
|
|
||||||
};
|
|
||||||
}
|
|
||||||
|
|
||||||
throw new Error('Attachment content must be a string or a buffer');
|
|
||||||
});
|
|
||||||
}
|
|
||||||
}
|
|
||||||
Reference in New Issue
Block a user