优化
This commit is contained in:
191
server/models/user/mail.ts
Normal file
191
server/models/user/mail.ts
Normal file
@@ -0,0 +1,191 @@
|
||||
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;
|
||||
Reference in New Issue
Block a user