You cannot select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
tailchat/server/models/user/mail.ts

192 lines
3.8 KiB
TypeScript

import {
getModelForClass,
prop,
DocumentType,
Ref,
modelOptions,
Severity,
ReturnModelType,
} from '@typegoose/typegoose';
import { Base, TimeStamps } from '@typegoose/typegoose/lib/defaultClasses';
import type { Types } from 'mongoose';
import { User } from './user';
import nodemailer, { Transporter, SendMailOptions } from 'nodemailer';
import { parseConnectionUrl } from 'nodemailer/lib/shared';
import { config } from 'tailchat-server-sdk';
import type SMTPConnection from 'nodemailer/lib/smtp-connection';
/**
* 将地址格式化
*/
function stringifyAddress(address: SendMailOptions['to']): string {
if (Array.isArray(address)) {
return address.map((a) => stringifyAddress(a)).join(',');
}
if (typeof address === 'string') {
return address;
} else if (address === undefined) {
return '';
} else if (typeof address === 'object') {
return `"${address.name}" ${address.address}`;
}
}
function getSMTPConnectionOptions(): SMTPConnection.Options | null {
if (config.smtp.connectionUrl) {
return parseConnectionUrl(config.smtp.connectionUrl);
}
return null;
}
@modelOptions({
options: {
allowMixed: Severity.ALLOW,
},
})
export class Mail extends TimeStamps implements Base {
_id: Types.ObjectId;
id: string;
/**
* 发件人邮箱
*/
@prop()
from: string;
/**
* 收件人邮箱
*/
@prop()
to: string;
/**
* 邮件主题
*/
@prop()
subject: string;
/**
* 邮件内容
*/
@prop()
body: string;
@prop()
host?: string;
@prop()
port?: string;
@prop()
secure?: boolean;
@prop()
is_success: boolean;
@prop()
data?: any;
@prop()
error?: string;
/**
* 创建邮件发送实例
*/
static createMailerTransporter(): Transporter | null {
const options = getSMTPConnectionOptions();
if (options) {
const transporter = nodemailer.createTransport(options);
return transporter;
}
return null;
}
/**
* 检查邮件服务是否可用
*/
static async verifyMailService(): Promise<boolean> {
try {
const transporter = Mail.createMailerTransporter();
if (!transporter) {
return false;
}
const verify = await transporter.verify();
return verify;
} catch (e) {
console.error(e);
return false;
}
}
/**
* 发送邮件
*/
static async sendMail(
this: ReturnModelType<typeof Mail>,
mailOptions: SendMailOptions
) {
try {
const transporter = Mail.createMailerTransporter();
if (!transporter) {
throw new Error('Mail Transporter is null');
}
const options = {
from: config.smtp.senderName,
...mailOptions,
};
const smtpOptions = getSMTPConnectionOptions();
try {
const info = await transporter.sendMail(options);
await this.create({
from: stringifyAddress(options.from),
to: stringifyAddress(options.to),
subject: options.subject,
body: options.html,
host: smtpOptions.host,
port: smtpOptions.port,
secure: smtpOptions.secure,
is_success: true,
data: info,
});
return info;
} catch (err) {
this.create({
from: stringifyAddress(options.from),
to: stringifyAddress(options.to),
subject: options.subject,
body: options.html,
host: smtpOptions.host,
port: smtpOptions.port,
secure: smtpOptions.secure,
is_success: false,
error: String(err),
});
throw err;
}
} catch (err) {
console.error(err);
throw err;
}
}
}
export type MailDocument = DocumentType<Mail>;
const model = getModelForClass(Mail);
export type MailModel = typeof model;
export default model;