mirror of
https://github.com/schroinerxy/cloud-mail.git
synced 2026-06-21 19:35:50 +08:00
614 lines
15 KiB
JavaScript
614 lines
15 KiB
JavaScript
import orm from '../entity/orm';
|
|
import email from '../entity/email';
|
|
import { attConst, emailConst, isDel, settingConst } from '../const/entity-const';
|
|
import { and, desc, eq, gt, inArray, lt, count, asc, sql, ne, or } from 'drizzle-orm';
|
|
import { star } from '../entity/star';
|
|
import settingService from './setting-service';
|
|
import accountService from './account-service';
|
|
import BizError from '../error/biz-error';
|
|
import emailUtils from '../utils/email-utils';
|
|
import { Resend } from 'resend';
|
|
import attService from './att-service';
|
|
import { parseHTML } from 'linkedom';
|
|
import userService from './user-service';
|
|
import roleService from './role-service';
|
|
import user from '../entity/user';
|
|
import starService from './star-service';
|
|
import dayjs from 'dayjs';
|
|
import kvConst from '../const/kv-const';
|
|
import { t } from '../i18n/i18n'
|
|
|
|
const emailService = {
|
|
|
|
async list(c, params, userId) {
|
|
|
|
let { emailId, type, accountId, size, timeSort } = params;
|
|
|
|
size = Number(size);
|
|
emailId = Number(emailId);
|
|
timeSort = Number(timeSort);
|
|
|
|
if (size > 30) {
|
|
size = 30;
|
|
}
|
|
|
|
if (!emailId) {
|
|
|
|
if (timeSort) {
|
|
emailId = 0;
|
|
} else {
|
|
emailId = 9999999999;
|
|
}
|
|
|
|
}
|
|
|
|
|
|
const query = orm(c)
|
|
.select({
|
|
...email,
|
|
starId: star.starId
|
|
})
|
|
.from(email)
|
|
.leftJoin(
|
|
star,
|
|
and(
|
|
eq(star.emailId, email.emailId),
|
|
eq(star.userId, userId)
|
|
)
|
|
)
|
|
.where(
|
|
and(
|
|
timeSort ? gt(email.emailId, emailId) : lt(email.emailId, emailId),
|
|
eq(email.accountId, accountId),
|
|
eq(email.userId, userId),
|
|
eq(email.type, type),
|
|
eq(email.isDel, isDel.NORMAL)
|
|
)
|
|
);
|
|
|
|
if (timeSort) {
|
|
query.orderBy(asc(email.emailId));
|
|
} else {
|
|
query.orderBy(desc(email.emailId));
|
|
}
|
|
|
|
const listQuery = query.limit(size).all();
|
|
|
|
const totalQuery = orm(c).select({ total: count() }).from(email).where(
|
|
and(
|
|
eq(email.accountId, accountId),
|
|
eq(email.userId, userId),
|
|
eq(email.type, type),
|
|
eq(email.isDel, isDel.NORMAL)
|
|
)
|
|
).get();
|
|
|
|
const latestEmailQuery = orm(c).select().from(email).where(
|
|
and(
|
|
eq(email.accountId, accountId),
|
|
eq(email.userId, userId),
|
|
eq(email.type, type),
|
|
eq(email.isDel, isDel.NORMAL)
|
|
))
|
|
.orderBy(desc(email.emailId)).limit(1).get();
|
|
|
|
let [list, totalRow, latestEmail] = await Promise.all([listQuery, totalQuery, latestEmailQuery]);
|
|
|
|
list = list.map(item => ({
|
|
...item,
|
|
isStar: item.starId != null ? 1 : 0
|
|
}));
|
|
|
|
const emailIds = list.map(item => item.emailId);
|
|
|
|
const attsList = await attService.selectByEmailIds(c, emailIds);
|
|
|
|
list.forEach(emailRow => {
|
|
const atts = attsList.filter(attsRow => attsRow.emailId === emailRow.emailId);
|
|
emailRow.attList = atts;
|
|
});
|
|
|
|
return { list, total: totalRow.total, latestEmail };
|
|
},
|
|
|
|
async delete(c, params, userId) {
|
|
const { emailIds } = params;
|
|
const emailIdList = emailIds.split(',').map(Number);
|
|
await orm(c).update(email).set({ isDel: isDel.DELETE }).where(
|
|
and(
|
|
eq(email.userId, userId),
|
|
inArray(email.emailId, emailIdList)))
|
|
.run();
|
|
},
|
|
|
|
receive(c, params, cidAttList, r2domain) {
|
|
params.content = this.imgReplace(params.content, cidAttList, r2domain)
|
|
return orm(c).insert(email).values({ ...params }).returning().get();
|
|
},
|
|
|
|
async send(c, params, userId) {
|
|
|
|
let { accountId, name, sendType, emailId, receiveEmail, manyType, text, content, subject, attachments } = params;
|
|
|
|
const { resendTokens, r2Domain, send } = await settingService.query(c);
|
|
|
|
let { attDataList, html } = await attService.toImageUrlHtml(c, content, r2Domain);
|
|
|
|
if (attDataList.length > 0 && !r2Domain) {
|
|
throw new BizError(t('noOsDomainSendPic'));
|
|
}
|
|
|
|
if (attDataList.length > 0 && !c.env.r2) {
|
|
throw new BizError(t('noOsSendPic'));
|
|
}
|
|
|
|
if (attachments.length > 0 && !r2Domain) {
|
|
throw new BizError(t('noOsDomainSendAtt'));
|
|
}
|
|
|
|
if (attachments.length > 0 && !c.env.r2) {
|
|
throw new BizError(t('noOsSendAtt'));
|
|
}
|
|
|
|
if (send === settingConst.send.CLOSE) {
|
|
throw new BizError(t('disabledSend'), 403);
|
|
}
|
|
|
|
if (attachments.length > 0 && manyType === 'divide') {
|
|
throw new BizError(t('noSeparateSend'));
|
|
}
|
|
|
|
|
|
const userRow = await userService.selectById(c, userId);
|
|
const roleRow = await roleService.selectById(c, userRow.type);
|
|
|
|
if (c.env.admin !== userRow.email && roleRow.sendCount) {
|
|
|
|
if (roleRow.sendCount < 0) {
|
|
throw new BizError(t('userNoSendTotal'), 403);
|
|
}
|
|
|
|
if (userRow.sendCount >= roleRow.sendCount) {
|
|
if (roleRow.sendType === 'day') throw new BizError(t('daySendLimit'), 403);
|
|
if (roleRow.sendType === 'count') throw new BizError(t('totalSendLimit'), 403);
|
|
}
|
|
|
|
if (userRow.sendCount + receiveEmail.length > roleRow.sendCount) {
|
|
if (roleRow.sendType === 'day') throw new BizError(t('daySendLack'), 403);
|
|
if (roleRow.sendType === 'count') throw new BizError(t('totalSendLack'), 403);
|
|
}
|
|
|
|
}
|
|
|
|
|
|
const accountRow = await accountService.selectById(c, accountId);
|
|
|
|
if (!accountRow) {
|
|
throw new BizError(t('senderAccountNotExist'));
|
|
}
|
|
|
|
const domain = emailUtils.getDomain(accountRow.email);
|
|
const resendToken = resendTokens[domain];
|
|
|
|
if (!resendToken) {
|
|
throw new BizError(t('noResendToken'));
|
|
}
|
|
|
|
|
|
if (accountRow.userId !== userId) {
|
|
throw new BizError(t('sendEmailNotCurUser'));
|
|
}
|
|
|
|
if (!name) {
|
|
name = emailUtils.getName(accountRow.email);
|
|
}
|
|
|
|
let emailRow = {
|
|
messageId: null
|
|
};
|
|
|
|
if (sendType === 'reply') {
|
|
|
|
emailRow = await this.selectById(c, emailId);
|
|
|
|
if (!emailRow) {
|
|
throw new BizError(t('notExistEmailReply'));
|
|
}
|
|
|
|
}
|
|
|
|
let resendResult = null;
|
|
|
|
const resend = new Resend(resendToken);
|
|
|
|
if (manyType === 'divide') {
|
|
|
|
let sendFormList = [];
|
|
|
|
receiveEmail.forEach(email => {
|
|
const sendForm = {
|
|
from: `${name} <${accountRow.email}>`,
|
|
to: [email],
|
|
subject: subject,
|
|
text: text,
|
|
html: html
|
|
};
|
|
|
|
if (sendType === 'reply') {
|
|
sendForm.headers = {
|
|
'in-reply-to': emailRow.messageId,
|
|
'references': emailRow.messageId
|
|
};
|
|
}
|
|
|
|
sendFormList.push(sendForm);
|
|
});
|
|
|
|
resendResult = await resend.batch.send(sendFormList);
|
|
|
|
} else {
|
|
|
|
const sendForm = {
|
|
from: `${name} <${accountRow.email}>`,
|
|
to: [...receiveEmail],
|
|
subject: subject,
|
|
text: text,
|
|
html: html,
|
|
attachments: attachments
|
|
};
|
|
|
|
if (sendType === 'reply') {
|
|
sendForm.headers = {
|
|
'in-reply-to': emailRow.messageId,
|
|
'references': emailRow.messageId
|
|
};
|
|
}
|
|
|
|
resendResult = await resend.emails.send(sendForm);
|
|
|
|
}
|
|
|
|
const { data, error } = resendResult;
|
|
|
|
|
|
if (error) {
|
|
console.error(error);
|
|
throw new BizError(error.message);
|
|
}
|
|
|
|
html = this.imgReplace(html, null, r2Domain);
|
|
|
|
const emailData = {};
|
|
emailData.sendEmail = accountRow.email;
|
|
emailData.name = name;
|
|
emailData.subject = subject;
|
|
emailData.content = html;
|
|
emailData.text = text;
|
|
emailData.accountId = accountId;
|
|
emailData.type = emailConst.type.SEND;
|
|
emailData.userId = userId;
|
|
emailData.status = emailConst.status.SENT;
|
|
|
|
const emailDataList = [];
|
|
|
|
if (manyType === 'divide') {
|
|
|
|
receiveEmail.forEach((item, index) => {
|
|
const emailDataItem = { ...emailData };
|
|
emailDataItem.resendEmailId = data.data[index].id;
|
|
emailDataItem.recipient = JSON.stringify([{ address: item, name: '' }]);
|
|
emailDataList.push(emailDataItem);
|
|
});
|
|
|
|
} else {
|
|
|
|
emailData.resendEmailId = data.id;
|
|
|
|
const recipient = [];
|
|
|
|
receiveEmail.forEach(item => {
|
|
recipient.push({ address: item, name: '' });
|
|
});
|
|
|
|
emailData.recipient = JSON.stringify(recipient);
|
|
|
|
emailDataList.push(emailData);
|
|
}
|
|
|
|
if (sendType === 'reply') {
|
|
emailDataList.forEach(emailData => {
|
|
emailData.inReplyTo = emailRow.messageId;
|
|
emailData.relation = emailRow.messageId;
|
|
});
|
|
}
|
|
|
|
|
|
if (roleRow.sendCount) {
|
|
await userService.incrUserSendCount(c, receiveEmail.length, userId);
|
|
}
|
|
|
|
const emailRowList = await Promise.all(
|
|
emailDataList.map(async (emailData) => {
|
|
const emailRow = await orm(c).insert(email).values(emailData).returning().get();
|
|
|
|
if (attDataList.length > 0) {
|
|
await attService.saveArticleAtt(c, attDataList, userId, accountId, emailRow.emailId);
|
|
}
|
|
|
|
if (attachments?.length > 0 && c.env.r2) {
|
|
await attService.saveSendAtt(c, attachments, userId, accountId, emailRow.emailId);
|
|
}
|
|
|
|
const attsList = await attService.selectByEmailIds(c, [emailRow.emailId]);
|
|
emailRow.attList = attsList;
|
|
|
|
return emailRow;
|
|
})
|
|
);
|
|
|
|
const dateStr = dayjs().format('YYYY-MM-DD');
|
|
|
|
let daySendTotal = await c.env.kv.get(kvConst.SEND_DAY_COUNT + dateStr);
|
|
|
|
if (!daySendTotal) {
|
|
await c.env.kv.put(kvConst.SEND_DAY_COUNT + dateStr, JSON.stringify(receiveEmail.length), { expirationTtl: 60 * 60 * 24 });
|
|
} else {
|
|
daySendTotal = Number(daySendTotal) + receiveEmail.length
|
|
await c.env.kv.put(kvConst.SEND_DAY_COUNT + dateStr, JSON.stringify(daySendTotal), { expirationTtl: 60 * 60 * 24 });
|
|
}
|
|
|
|
return emailRowList;
|
|
},
|
|
|
|
imgReplace(content, cidAttList, r2domain) {
|
|
|
|
if (!content) {
|
|
return ''
|
|
}
|
|
|
|
const { document } = parseHTML(content);
|
|
|
|
const images = Array.from(document.querySelectorAll('img'));
|
|
|
|
const useAtts = []
|
|
|
|
for (const img of images) {
|
|
|
|
const src = img.getAttribute('src');
|
|
if (src && src.startsWith('cid:') && cidAttList) {
|
|
|
|
const cid = src.replace(/^cid:/, '');
|
|
const attCidIndex = cidAttList.findIndex(cidAtt => cidAtt.contentId.replace(/^<|>$/g, '') === cid);
|
|
|
|
if (attCidIndex > -1) {
|
|
const cidAtt = cidAttList[attCidIndex];
|
|
img.setAttribute('src', '{{domain}}' + cidAtt.key);
|
|
useAtts.push(cidAtt)
|
|
}
|
|
|
|
}
|
|
|
|
if (src && src.startsWith(r2domain + '/')) {
|
|
img.setAttribute('src', src.replace(r2domain + '/', '{{domain}}'));
|
|
}
|
|
|
|
}
|
|
|
|
useAtts.forEach(att => {
|
|
att.type = attConst.type.EMBED
|
|
})
|
|
|
|
return document.toString();
|
|
},
|
|
|
|
selectById(c, emailId) {
|
|
return orm(c).select().from(email).where(
|
|
and(eq(email.emailId, emailId),
|
|
eq(email.isDel, isDel.NORMAL)))
|
|
.get();
|
|
},
|
|
|
|
async latest(c, params, userId) {
|
|
let { emailId, accountId } = params;
|
|
const list = await orm(c).select().from(email).where(
|
|
and(
|
|
eq(email.userId, userId),
|
|
eq(email.isDel, isDel.NORMAL),
|
|
eq(email.accountId, accountId),
|
|
eq(email.type, emailConst.type.RECEIVE),
|
|
gt(email.emailId, emailId)
|
|
))
|
|
.orderBy(desc(email.emailId))
|
|
.limit(20);
|
|
|
|
const emailIds = list.map(item => item.emailId);
|
|
|
|
if (emailIds.length > 0) {
|
|
|
|
const attsList = await attService.selectByEmailIds(c, emailIds);
|
|
|
|
list.forEach(emailRow => {
|
|
const atts = attsList.filter(attsRow => attsRow.emailId === emailRow.emailId);
|
|
emailRow.attList = atts;
|
|
});
|
|
}
|
|
|
|
return list;
|
|
},
|
|
|
|
|
|
async physicsDeleteAll(c) {
|
|
const emailIdsRow = await orm(c).select({ emailId: email.emailId }).from(email).where(eq(email.isDel, isDel.DELETE)).limit(99);
|
|
if (emailIdsRow.length === 0) {
|
|
return;
|
|
}
|
|
const emailIds = emailIdsRow.map(row => row.emailId);
|
|
await attService.removeByEmailIds(c, emailIds);
|
|
await starService.removeByEmailIds(c, emailIds);
|
|
await orm(c).delete(email).where(inArray(email.emailId, emailIds)).run();
|
|
if (emailIdsRow.length === 99) {
|
|
await this.physicsDeleteAll(c);
|
|
}
|
|
},
|
|
|
|
async physicsDelete(c, params) {
|
|
let { emailIds } = params;
|
|
emailIds = emailIds.split(',').map(Number);
|
|
await attService.removeByEmailIds(c, emailIds);
|
|
await starService.removeByEmailIds(c, emailIds);
|
|
await orm(c).delete(email).where(inArray(email.emailId, emailIds)).run();
|
|
},
|
|
|
|
async physicsDeleteAccountIds(c, accountIds) {
|
|
await attService.removeByAccountIds(c, accountIds);
|
|
await orm(c).delete(email).where(inArray(email.accountId, accountIds)).run();
|
|
},
|
|
|
|
async physicsDeleteUserIds(c, userIds) {
|
|
await attService.removeByUserIds(c, userIds);
|
|
await orm(c).delete(email).where(inArray(email.userId, userIds)).run();
|
|
},
|
|
|
|
updateEmailStatus(c, params) {
|
|
const { status, resendEmailId, message } = params;
|
|
return orm(c).update(email).set({
|
|
status: status,
|
|
message: message
|
|
}).where(eq(email.resendEmailId, resendEmailId)).returning().get();
|
|
},
|
|
|
|
async selectUserEmailCountList(c, userIds, type, del = isDel.NORMAL) {
|
|
const result = await orm(c)
|
|
.select({
|
|
userId: email.userId,
|
|
count: count(email.emailId)
|
|
})
|
|
.from(email)
|
|
.where(and(
|
|
inArray(email.userId, userIds),
|
|
eq(email.type, type),
|
|
eq(email.isDel, del)
|
|
))
|
|
.groupBy(email.userId);
|
|
return result;
|
|
},
|
|
|
|
async allList(c, params) {
|
|
|
|
let { emailId, size, name, subject, accountEmail, userEmail, type, timeSort } = params;
|
|
|
|
size = Number(size);
|
|
|
|
emailId = Number(emailId);
|
|
timeSort = Number(timeSort);
|
|
|
|
if (size > 30) {
|
|
size = 30;
|
|
}
|
|
|
|
if (!emailId) {
|
|
|
|
if (timeSort) {
|
|
emailId = 0;
|
|
} else {
|
|
emailId = 9999999999;
|
|
}
|
|
|
|
}
|
|
|
|
const conditions = [];
|
|
|
|
|
|
if (type === 'send') {
|
|
conditions.push(eq(email.type, emailConst.type.SEND));
|
|
}
|
|
|
|
if (type === 'receive') {
|
|
conditions.push(eq(email.type, emailConst.type.RECEIVE));
|
|
}
|
|
|
|
if (type === 'delete') {
|
|
conditions.push(eq(email.isDel, isDel.DELETE));
|
|
}
|
|
|
|
if (type === 'noone') {
|
|
conditions.push(eq(email.status, emailConst.status.NOONE));
|
|
}
|
|
|
|
if (userEmail) {
|
|
conditions.push(sql`${user.email} COLLATE NOCASE LIKE ${userEmail + '%'}`);
|
|
}
|
|
|
|
if (accountEmail) {
|
|
conditions.push(
|
|
or(
|
|
sql`${email.toEmail} COLLATE NOCASE LIKE ${accountEmail + '%'}`,
|
|
sql`${email.sendEmail} COLLATE NOCASE LIKE ${accountEmail + '%'}`,
|
|
)
|
|
)
|
|
}
|
|
|
|
if (name) {
|
|
conditions.push(sql`${email.name} COLLATE NOCASE LIKE ${name + '%'}`);
|
|
}
|
|
|
|
if (subject) {
|
|
conditions.push(sql`${email.subject} COLLATE NOCASE LIKE ${subject + '%'}`);
|
|
}
|
|
|
|
conditions.push(ne(email.status, emailConst.status.SAVING));
|
|
|
|
const countConditions = [...conditions];
|
|
|
|
if (timeSort) {
|
|
conditions.push(gt(email.emailId, emailId));
|
|
} else {
|
|
conditions.push(lt(email.emailId, emailId));
|
|
}
|
|
|
|
const query = orm(c).select({ ...email, userEmail: user.email })
|
|
.from(email)
|
|
.leftJoin(user, eq(email.userId, user.userId))
|
|
.where(and(...conditions));
|
|
|
|
const queryCount = orm(c).select({ total: count() })
|
|
.from(email)
|
|
.leftJoin(user, eq(email.userId, user.userId))
|
|
.where(and(...countConditions));
|
|
|
|
if (timeSort) {
|
|
query.orderBy(asc(email.emailId));
|
|
} else {
|
|
query.orderBy(desc(email.emailId));
|
|
}
|
|
|
|
const listQuery = await query.limit(size).all();
|
|
const totalQuery = await queryCount.get();
|
|
|
|
const [list, totalRow] = await Promise.all([listQuery, totalQuery]);
|
|
|
|
const emailIds = list.map(item => item.emailId);
|
|
const attsList = await attService.selectByEmailIds(c, emailIds);
|
|
|
|
list.forEach(emailRow => {
|
|
const atts = attsList.filter(attsRow => attsRow.emailId === emailRow.emailId);
|
|
emailRow.attList = atts;
|
|
});
|
|
|
|
return { list: list, total: totalRow.total };
|
|
},
|
|
|
|
async restoreByUserId(c, userId) {
|
|
await orm(c).update(email).set({ isDel: isDel.NORMAL }).where(eq(email.userId, userId)).run();
|
|
},
|
|
|
|
async completeReceive(c, status, emailId) {
|
|
return await orm(c).update(email).set({
|
|
isDel: isDel.NORMAL,
|
|
status: status
|
|
}).where(eq(email.emailId, emailId)).returning().get();
|
|
}
|
|
};
|
|
|
|
export default emailService;
|