mirror of
https://github.com/nodemailer/wildduck.git
synced 2025-01-11 02:18:38 +08:00
410 lines
17 KiB
JavaScript
410 lines
17 KiB
JavaScript
'use strict';
|
|
|
|
const imapTools = require('../../imap-core/lib/imap-tools');
|
|
const db = require('../db');
|
|
const tools = require('../tools');
|
|
const consts = require('../consts');
|
|
const base32 = require('base32.js');
|
|
const crypto = require('crypto');
|
|
|
|
// STORE / UID STORE, updates flags for selected UIDs
|
|
module.exports = server => (path, update, session, callback) => {
|
|
server.logger.debug(
|
|
{
|
|
tnx: 'store',
|
|
cid: session.id
|
|
},
|
|
'[%s] Updating messages in "%s"',
|
|
session.id,
|
|
path
|
|
);
|
|
db.database.collection('mailboxes').findOne(
|
|
{
|
|
user: session.user.id,
|
|
path
|
|
},
|
|
(err, mailboxData) => {
|
|
if (err) {
|
|
return callback(err);
|
|
}
|
|
|
|
if (!mailboxData) {
|
|
return callback(null, 'NONEXISTENT');
|
|
}
|
|
|
|
let unseenChange = update.value.includes('\\Seen');
|
|
|
|
let query = {
|
|
mailbox: mailboxData._id
|
|
};
|
|
|
|
if (update.unchangedSince) {
|
|
query = {
|
|
mailbox: mailboxData._id,
|
|
modseq: {
|
|
$lte: update.unchangedSince
|
|
}
|
|
};
|
|
}
|
|
|
|
let queryAll = false;
|
|
if (update.messages.length !== session.selected.uidList.length) {
|
|
// do not use uid selector for 1:*
|
|
query.uid = tools.checkRangeQuery(update.messages);
|
|
} else {
|
|
// 1:*
|
|
queryAll = true;
|
|
// uid is part of the sharding key so we need it somehow represented in the query
|
|
query.uid = {
|
|
$gt: 0,
|
|
$lt: mailboxData.uidNext
|
|
};
|
|
}
|
|
|
|
let cursor = db.database
|
|
.collection('messages')
|
|
.find(query)
|
|
.project({
|
|
_id: true,
|
|
uid: true,
|
|
flags: true
|
|
})
|
|
.sort([['uid', 1]]);
|
|
|
|
let shouldExpunge = false;
|
|
let updateEntries = [];
|
|
let notifyEntries = [];
|
|
|
|
let done = (...args) => {
|
|
if (shouldExpunge) {
|
|
// shcedule EXPUNGE command for current folder
|
|
let expungeOptions = {
|
|
// create new temporary session so it would not mix with the active one
|
|
id: 'auto.' + base32.encode(crypto.randomBytes(10)).toLowerCase(),
|
|
user: {
|
|
id: session.user.id,
|
|
username: session.user.username
|
|
}
|
|
};
|
|
setImmediate(() => {
|
|
server.onExpunge(path, { silent: true }, expungeOptions, () => false);
|
|
});
|
|
}
|
|
|
|
if (updateEntries.length) {
|
|
return db.database.collection('messages').bulkWrite(
|
|
updateEntries,
|
|
{
|
|
ordered: false,
|
|
w: 1
|
|
},
|
|
() => {
|
|
updateEntries = [];
|
|
server.notifier.addEntries(session.user.id, path, notifyEntries, () => {
|
|
notifyEntries = [];
|
|
server.notifier.fire(session.user.id, path);
|
|
if (args[0]) {
|
|
// first argument is an error
|
|
return callback(...args);
|
|
} else {
|
|
updateMailboxFlags(mailboxData, update, () => callback(...args));
|
|
}
|
|
});
|
|
}
|
|
);
|
|
}
|
|
server.notifier.fire(session.user.id, path);
|
|
if (args[0]) {
|
|
// first argument is an error
|
|
return callback(...args);
|
|
} else {
|
|
updateMailboxFlags(mailboxData, update, () => callback(...args));
|
|
}
|
|
};
|
|
|
|
// We have to process all messages one by one instead of just calling an update
|
|
// for all messages as we need to know which messages were exactly modified,
|
|
// otherwise we can't send flag update notifications and modify modseq values
|
|
let processNext = () => {
|
|
cursor.next((err, message) => {
|
|
if (err) {
|
|
return done(err);
|
|
}
|
|
if (!message) {
|
|
return cursor.close(() => done(null, true));
|
|
}
|
|
if (queryAll && !session.selected.uidList.includes(message.uid)) {
|
|
// skip processing messages that we do not know about yet
|
|
return processNext();
|
|
}
|
|
|
|
let flagsupdate = false; // query object for updates
|
|
|
|
let updated = false;
|
|
let existingFlags = message.flags.map(flag => flag.toLowerCase().trim());
|
|
switch (update.action) {
|
|
case 'set':
|
|
// check if update set matches current or is different
|
|
if (
|
|
// if length does not match
|
|
existingFlags.length !== update.value.length ||
|
|
// or a new flag was found
|
|
update.value.filter(flag => !existingFlags.includes(flag.toLowerCase().trim())).length
|
|
) {
|
|
updated = true;
|
|
}
|
|
|
|
message.flags = [].concat(update.value);
|
|
|
|
// set flags
|
|
if (updated) {
|
|
flagsupdate = {
|
|
$set: {
|
|
flags: message.flags,
|
|
unseen: !message.flags.includes('\\Seen'),
|
|
flagged: message.flags.includes('\\Flagged'),
|
|
undeleted: !message.flags.includes('\\Deleted'),
|
|
draft: message.flags.includes('\\Draft')
|
|
}
|
|
};
|
|
|
|
if (message.flags.includes('\\Deleted')) {
|
|
shouldExpunge = true;
|
|
}
|
|
|
|
if (!['\\Junk', '\\Trash'].includes(mailboxData.specialUse) && !message.flags.includes('\\Deleted')) {
|
|
flagsupdate.$set.searchable = true;
|
|
} else {
|
|
flagsupdate.$unset = {
|
|
searchable: ''
|
|
};
|
|
}
|
|
}
|
|
break;
|
|
|
|
case 'add': {
|
|
let newFlags = [];
|
|
message.flags = message.flags.concat(
|
|
update.value.filter(flag => {
|
|
if (!existingFlags.includes(flag.toLowerCase().trim())) {
|
|
updated = true;
|
|
newFlags.push(flag);
|
|
return true;
|
|
}
|
|
return false;
|
|
})
|
|
);
|
|
|
|
// add flags
|
|
if (updated) {
|
|
flagsupdate = {
|
|
$addToSet: {
|
|
flags: {
|
|
$each: newFlags
|
|
}
|
|
}
|
|
};
|
|
|
|
if (
|
|
newFlags.includes('\\Seen') ||
|
|
newFlags.includes('\\Flagged') ||
|
|
newFlags.includes('\\Deleted') ||
|
|
newFlags.includes('\\Draft')
|
|
) {
|
|
flagsupdate.$set = {};
|
|
if (newFlags.includes('\\Seen')) {
|
|
flagsupdate.$set = {
|
|
unseen: false
|
|
};
|
|
}
|
|
if (newFlags.includes('\\Flagged')) {
|
|
flagsupdate.$set = {
|
|
flagged: true
|
|
};
|
|
}
|
|
if (newFlags.includes('\\Deleted')) {
|
|
shouldExpunge = true;
|
|
flagsupdate.$set = {
|
|
undeleted: false
|
|
};
|
|
flagsupdate.$unset = {
|
|
searchable: ''
|
|
};
|
|
}
|
|
if (newFlags.includes('\\Draft')) {
|
|
flagsupdate.$set = {
|
|
draft: true
|
|
};
|
|
}
|
|
}
|
|
}
|
|
break;
|
|
}
|
|
|
|
case 'remove': {
|
|
// We need to use the case of existing flags when removing
|
|
let oldFlags = [];
|
|
let flagsUpdates = update.value.map(flag => flag.toLowerCase().trim());
|
|
message.flags = message.flags.filter(flag => {
|
|
if (!flagsUpdates.includes(flag.toLowerCase().trim())) {
|
|
return true;
|
|
}
|
|
oldFlags.push(flag);
|
|
updated = true;
|
|
return false;
|
|
});
|
|
|
|
// remove flags
|
|
if (updated) {
|
|
flagsupdate = {
|
|
$pull: {
|
|
flags: {
|
|
$in: oldFlags
|
|
}
|
|
}
|
|
};
|
|
if (
|
|
oldFlags.includes('\\Seen') ||
|
|
oldFlags.includes('\\Flagged') ||
|
|
oldFlags.includes('\\Deleted') ||
|
|
oldFlags.includes('\\Draft')
|
|
) {
|
|
flagsupdate.$set = {};
|
|
if (oldFlags.includes('\\Seen')) {
|
|
flagsupdate.$set = {
|
|
unseen: true
|
|
};
|
|
}
|
|
if (oldFlags.includes('\\Flagged')) {
|
|
flagsupdate.$set = {
|
|
flagged: false
|
|
};
|
|
}
|
|
if (oldFlags.includes('\\Deleted')) {
|
|
flagsupdate.$set = {
|
|
undeleted: true
|
|
};
|
|
if (!['\\Junk', '\\Trash'].includes(mailboxData.specialUse)) {
|
|
flagsupdate.$set.searchable = true;
|
|
}
|
|
}
|
|
if (oldFlags.includes('\\Draft')) {
|
|
flagsupdate.$set = {
|
|
draft: false
|
|
};
|
|
}
|
|
}
|
|
}
|
|
break;
|
|
}
|
|
}
|
|
|
|
if (!update.silent) {
|
|
// print updated state of the message
|
|
session.writeStream.write(
|
|
session.formatResponse('FETCH', message.uid, {
|
|
uid: update.isUid ? message.uid : false,
|
|
flags: message.flags
|
|
})
|
|
);
|
|
}
|
|
|
|
if (updated) {
|
|
updateEntries.push({
|
|
updateOne: {
|
|
filter: {
|
|
_id: message._id,
|
|
// include shard key data as well
|
|
mailbox: mailboxData._id,
|
|
uid: message.uid
|
|
},
|
|
update: flagsupdate
|
|
}
|
|
});
|
|
|
|
notifyEntries.push({
|
|
command: 'FETCH',
|
|
ignore: session.id,
|
|
uid: message.uid,
|
|
flags: message.flags,
|
|
message: message._id,
|
|
unseenChange
|
|
});
|
|
|
|
if (updateEntries.length >= consts.BULK_BATCH_SIZE) {
|
|
return db.database.collection('messages').bulkWrite(
|
|
updateEntries,
|
|
{
|
|
ordered: false,
|
|
w: 1
|
|
},
|
|
err => {
|
|
updateEntries = [];
|
|
if (err) {
|
|
return cursor.close(() => done(err));
|
|
}
|
|
|
|
server.notifier.addEntries(session.user.id, path, notifyEntries, () => {
|
|
notifyEntries = [];
|
|
server.notifier.fire(session.user.id, path);
|
|
processNext();
|
|
});
|
|
}
|
|
);
|
|
} else {
|
|
processNext();
|
|
}
|
|
} else {
|
|
processNext();
|
|
}
|
|
});
|
|
};
|
|
|
|
processNext();
|
|
}
|
|
);
|
|
};
|
|
|
|
function updateMailboxFlags(mailbox, update, callback) {
|
|
if (update.action === 'remove') {
|
|
// we didn't add any new flags, so there's nothing to update
|
|
return callback();
|
|
}
|
|
|
|
let mailboxFlags = imapTools.systemFlags.concat(mailbox.flags || []).map(flag => flag.trim().toLowerCase());
|
|
let newFlags = [];
|
|
|
|
// find flags that are not listed with mailbox
|
|
update.value.forEach(flag => {
|
|
// limit mailbox flags by 100
|
|
if (mailboxFlags.length + newFlags.length >= 100) {
|
|
return;
|
|
}
|
|
// if mailbox does not have such flag, then add it
|
|
if (!mailboxFlags.includes(flag.toLowerCase().trim())) {
|
|
newFlags.push(flag);
|
|
}
|
|
});
|
|
|
|
// nothing new found
|
|
if (!newFlags.length) {
|
|
return callback();
|
|
}
|
|
|
|
// found some new flags not yet set for mailbox
|
|
// FIXME: Should we send unsolicited FLAGS and PERMANENTFLAGS notifications? Probably not
|
|
return db.database.collection('mailboxes').findOneAndUpdate(
|
|
{
|
|
_id: mailbox._id
|
|
},
|
|
{
|
|
$addToSet: {
|
|
flags: {
|
|
$each: newFlags
|
|
}
|
|
}
|
|
},
|
|
{},
|
|
callback
|
|
);
|
|
}
|