2017-07-16 00:08:33 +08:00
|
|
|
'use strict';
|
|
|
|
|
|
|
|
const db = require('../db');
|
2017-12-15 17:02:47 +08:00
|
|
|
const tools = require('../tools');
|
2017-07-16 00:08:33 +08:00
|
|
|
|
|
|
|
// EXPUNGE deletes all messages in selected mailbox marked with \Delete
|
2017-12-10 07:19:50 +08:00
|
|
|
module.exports = (server, messageHandler) => (mailbox, update, session, callback) => {
|
2017-07-16 00:08:33 +08:00
|
|
|
server.logger.debug(
|
|
|
|
{
|
|
|
|
tnx: 'expunge',
|
|
|
|
cid: session.id
|
|
|
|
},
|
|
|
|
'[%s] Deleting messages from "%s"',
|
|
|
|
session.id,
|
2017-12-10 07:19:50 +08:00
|
|
|
mailbox
|
2017-07-16 00:08:33 +08:00
|
|
|
);
|
2017-11-22 22:22:36 +08:00
|
|
|
db.database.collection('mailboxes').findOne(
|
|
|
|
{
|
2017-12-10 07:19:50 +08:00
|
|
|
_id: mailbox
|
2017-11-22 22:22:36 +08:00
|
|
|
},
|
|
|
|
(err, mailboxData) => {
|
|
|
|
if (err) {
|
|
|
|
return callback(err);
|
|
|
|
}
|
|
|
|
if (!mailboxData) {
|
|
|
|
return callback(null, 'NONEXISTENT');
|
2017-07-16 00:08:33 +08:00
|
|
|
}
|
|
|
|
|
2017-12-15 17:02:47 +08:00
|
|
|
let query = {
|
|
|
|
user: session.user.id,
|
|
|
|
mailbox: mailboxData._id,
|
|
|
|
undeleted: false,
|
|
|
|
// uid is part of the sharding key so we need it somehow represented in the query
|
|
|
|
uid: {}
|
|
|
|
};
|
|
|
|
|
|
|
|
if (update.isUid) {
|
|
|
|
query.uid = tools.checkRangeQuery(update.messages);
|
|
|
|
} else {
|
|
|
|
query.uid.$gt = 0;
|
|
|
|
query.uid.$lt = mailboxData.uidNext;
|
|
|
|
}
|
|
|
|
|
2017-11-22 22:22:36 +08:00
|
|
|
let cursor = db.database
|
|
|
|
.collection('messages')
|
2017-12-15 17:02:47 +08:00
|
|
|
.find(query)
|
2017-11-22 22:22:36 +08:00
|
|
|
.sort([['uid', 1]]);
|
2017-07-16 00:08:33 +08:00
|
|
|
|
2017-11-22 22:22:36 +08:00
|
|
|
let deletedMessages = 0;
|
|
|
|
let deletedStorage = 0;
|
|
|
|
|
|
|
|
let updateQuota = next => {
|
|
|
|
if (!deletedMessages) {
|
|
|
|
return next();
|
2017-07-16 00:08:33 +08:00
|
|
|
}
|
|
|
|
|
2017-11-22 22:22:36 +08:00
|
|
|
db.users.collection('users').findOneAndUpdate(
|
|
|
|
{
|
|
|
|
_id: mailboxData.user
|
|
|
|
},
|
|
|
|
{
|
|
|
|
$inc: {
|
|
|
|
storageUsed: -deletedStorage
|
|
|
|
}
|
|
|
|
},
|
|
|
|
next
|
|
|
|
);
|
|
|
|
};
|
|
|
|
|
|
|
|
let processNext = () => {
|
|
|
|
cursor.next((err, messageData) => {
|
2017-07-16 00:08:33 +08:00
|
|
|
if (err) {
|
2017-11-22 22:22:36 +08:00
|
|
|
return updateQuota(() => callback(err));
|
2017-07-16 00:08:33 +08:00
|
|
|
}
|
2017-11-22 22:22:36 +08:00
|
|
|
if (!messageData) {
|
|
|
|
return cursor.close(() => {
|
|
|
|
updateQuota(() => {
|
2017-12-10 07:19:50 +08:00
|
|
|
server.notifier.fire(session.user.id);
|
2017-11-22 22:22:36 +08:00
|
|
|
if (!update.silent && session && session.selected && session.selected.uidList) {
|
|
|
|
session.writeStream.write({
|
|
|
|
tag: '*',
|
|
|
|
command: String(session.selected.uidList.length),
|
|
|
|
attributes: [
|
|
|
|
{
|
|
|
|
type: 'atom',
|
|
|
|
value: 'EXISTS'
|
|
|
|
}
|
|
|
|
]
|
|
|
|
});
|
|
|
|
}
|
|
|
|
return callback(null, true);
|
|
|
|
});
|
|
|
|
});
|
2017-07-16 00:08:33 +08:00
|
|
|
}
|
|
|
|
|
2017-11-22 22:22:36 +08:00
|
|
|
messageHandler.del(
|
|
|
|
{
|
|
|
|
messageData,
|
|
|
|
session,
|
|
|
|
// do not archive drafts
|
|
|
|
archive: !messageData.flags.includes('\\Draft'),
|
|
|
|
delayNotifications: true
|
|
|
|
},
|
|
|
|
err => {
|
|
|
|
if (err) {
|
|
|
|
server.logger.error(
|
|
|
|
{
|
|
|
|
tnx: 'EXPUNGE',
|
|
|
|
err
|
|
|
|
},
|
|
|
|
'Failed to delete message id=%s. %s',
|
|
|
|
messageData._id,
|
|
|
|
err.message
|
|
|
|
);
|
|
|
|
return cursor.close(() => updateQuota(() => callback(err)));
|
|
|
|
}
|
|
|
|
server.logger.debug(
|
|
|
|
{
|
|
|
|
tnx: 'EXPUNGE',
|
|
|
|
err
|
|
|
|
},
|
|
|
|
'Deleted message id=%s',
|
|
|
|
messageData._id
|
|
|
|
);
|
|
|
|
deletedMessages++;
|
|
|
|
deletedStorage += Number(messageData.size) || 0;
|
2017-11-17 19:37:53 +08:00
|
|
|
|
2017-11-22 22:22:36 +08:00
|
|
|
if (!update.silent) {
|
|
|
|
session.writeStream.write(session.formatResponse('EXPUNGE', messageData.uid));
|
|
|
|
}
|
2017-11-17 19:37:53 +08:00
|
|
|
|
2017-11-22 22:22:36 +08:00
|
|
|
setImmediate(processNext);
|
|
|
|
}
|
|
|
|
);
|
2017-07-16 00:08:33 +08:00
|
|
|
});
|
2017-11-22 22:22:36 +08:00
|
|
|
};
|
2017-07-16 00:08:33 +08:00
|
|
|
|
2017-11-22 22:22:36 +08:00
|
|
|
processNext();
|
|
|
|
}
|
|
|
|
);
|
2017-07-16 00:08:33 +08:00
|
|
|
};
|