2017-07-16 00:08:33 +08:00
|
|
|
'use strict';
|
|
|
|
|
|
|
|
const ObjectID = require('mongodb').ObjectID;
|
|
|
|
const db = require('../db');
|
2017-07-17 03:40:34 +08:00
|
|
|
const tools = require('../tools');
|
2017-07-16 00:08:33 +08:00
|
|
|
|
|
|
|
// COPY / UID COPY sequence mailbox
|
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: 'copy',
|
|
|
|
cid: session.id
|
|
|
|
},
|
|
|
|
'[%s] Copying messages from "%s" to "%s"',
|
|
|
|
session.id,
|
2017-12-10 07:19:50 +08:00
|
|
|
mailbox,
|
2017-07-16 00:08:33 +08:00
|
|
|
update.destination
|
|
|
|
);
|
2017-08-07 02:25:10 +08:00
|
|
|
|
2017-12-10 07:19:50 +08:00
|
|
|
db.database.collection('mailboxes').findOne(
|
|
|
|
{
|
|
|
|
_id: mailbox
|
|
|
|
},
|
|
|
|
(err, mailboxData) => {
|
2017-07-16 00:08:33 +08:00
|
|
|
if (err) {
|
|
|
|
return callback(err);
|
|
|
|
}
|
2017-12-10 07:19:50 +08:00
|
|
|
if (!mailboxData) {
|
|
|
|
return callback(null, 'NONEXISTENT');
|
2017-07-16 00:08:33 +08:00
|
|
|
}
|
|
|
|
|
2017-12-10 07:19:50 +08:00
|
|
|
db.database.collection('mailboxes').findOne(
|
|
|
|
{
|
|
|
|
user: session.user.id,
|
|
|
|
path: update.destination
|
|
|
|
},
|
|
|
|
(err, targetData) => {
|
2017-07-16 00:08:33 +08:00
|
|
|
if (err) {
|
2017-12-10 07:19:50 +08:00
|
|
|
return callback(err);
|
2017-07-16 00:08:33 +08:00
|
|
|
}
|
2017-12-10 07:19:50 +08:00
|
|
|
if (!targetData) {
|
|
|
|
return callback(null, 'TRYCREATE');
|
2017-07-16 00:08:33 +08:00
|
|
|
}
|
|
|
|
|
2017-12-10 07:19:50 +08:00
|
|
|
let cursor = db.database
|
|
|
|
.collection('messages')
|
|
|
|
.find({
|
|
|
|
mailbox: mailboxData._id,
|
|
|
|
uid: tools.checkRangeQuery(update.messages)
|
|
|
|
})
|
|
|
|
.sort([['uid', 1]]); // no projection as we need to copy the entire message
|
2017-07-16 00:08:33 +08:00
|
|
|
|
2017-12-10 07:19:50 +08:00
|
|
|
let copiedMessages = 0;
|
|
|
|
let copiedStorage = 0;
|
2017-10-20 18:43:44 +08:00
|
|
|
|
2017-12-10 07:19:50 +08:00
|
|
|
let updateQuota = next => {
|
|
|
|
if (!copiedMessages) {
|
|
|
|
return next();
|
2017-10-20 18:43:44 +08:00
|
|
|
}
|
2017-12-10 07:19:50 +08:00
|
|
|
db.users.collection('users').findOneAndUpdate(
|
|
|
|
{
|
|
|
|
_id: mailboxData.user
|
|
|
|
},
|
|
|
|
{
|
|
|
|
$inc: {
|
|
|
|
storageUsed: copiedStorage
|
|
|
|
}
|
|
|
|
},
|
|
|
|
next
|
|
|
|
);
|
|
|
|
};
|
|
|
|
|
|
|
|
let sourceUid = [];
|
|
|
|
let destinationUid = [];
|
|
|
|
let processNext = () => {
|
|
|
|
cursor.next((err, message) => {
|
2017-07-16 00:08:33 +08:00
|
|
|
if (err) {
|
2017-12-10 07:19:50 +08:00
|
|
|
return updateQuota(() => callback(err));
|
|
|
|
}
|
|
|
|
if (!message) {
|
2017-07-16 00:08:33 +08:00
|
|
|
return cursor.close(() => {
|
2017-12-10 07:19:50 +08:00
|
|
|
updateQuota(() => {
|
|
|
|
server.notifier.fire(session.user.id, targetData.path);
|
|
|
|
return callback(null, true, {
|
|
|
|
uidValidity: targetData.uidValidity,
|
|
|
|
sourceUid,
|
|
|
|
destinationUid
|
|
|
|
});
|
|
|
|
});
|
2017-07-16 00:08:33 +08:00
|
|
|
});
|
|
|
|
}
|
|
|
|
|
2017-12-10 07:19:50 +08:00
|
|
|
// Copying is not done in bulk to minimize risk of going out of sync with incremental UIDs
|
|
|
|
sourceUid.unshift(message.uid);
|
|
|
|
db.database.collection('mailboxes').findOneAndUpdate(
|
|
|
|
{
|
|
|
|
_id: targetData._id
|
|
|
|
},
|
|
|
|
{
|
|
|
|
$inc: {
|
|
|
|
uidNext: 1
|
|
|
|
}
|
|
|
|
},
|
|
|
|
{
|
|
|
|
uidNext: true
|
|
|
|
},
|
|
|
|
(err, item) => {
|
|
|
|
if (err) {
|
|
|
|
return cursor.close(() => {
|
|
|
|
updateQuota(() => callback(err));
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!item || !item.value) {
|
|
|
|
// was not able to acquire a lock
|
|
|
|
return cursor.close(() => {
|
|
|
|
updateQuota(() => callback(null, 'TRYCREATE'));
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
|
|
|
let uidNext = item.value.uidNext;
|
|
|
|
destinationUid.unshift(uidNext);
|
|
|
|
|
|
|
|
message._id = new ObjectID();
|
|
|
|
message.mailbox = targetData._id;
|
|
|
|
message.uid = uidNext;
|
|
|
|
|
|
|
|
// retention settings
|
|
|
|
message.exp = !!targetData.retention;
|
|
|
|
message.rdate = Date.now() + (targetData.retention || 0);
|
|
|
|
|
|
|
|
if (['\\Junk', '\\Trash'].includes(targetData.specialUse)) {
|
|
|
|
delete message.searchable;
|
|
|
|
} else {
|
|
|
|
message.searchable = true;
|
|
|
|
}
|
|
|
|
|
|
|
|
let junk = false;
|
|
|
|
if (targetData.specialUse === '\\Junk' && !message.junk) {
|
|
|
|
message.junk = true;
|
|
|
|
junk = 1;
|
|
|
|
} else if (targetData.specialUse !== '\\Trash' && message.junk) {
|
|
|
|
delete message.junk;
|
|
|
|
junk = -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!message.meta) {
|
|
|
|
message.meta = {};
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!message.meta.events) {
|
|
|
|
message.meta.events = [];
|
|
|
|
}
|
|
|
|
message.meta.events.push({
|
|
|
|
action: 'IMAPCOPY',
|
|
|
|
time: new Date()
|
|
|
|
});
|
|
|
|
|
|
|
|
db.database.collection('messages').insertOne(message, err => {
|
|
|
|
if (err) {
|
|
|
|
return cursor.close(() => {
|
|
|
|
updateQuota(() => callback(err));
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
|
|
|
copiedMessages++;
|
|
|
|
copiedStorage += Number(message.size) || 0;
|
|
|
|
|
|
|
|
let attachmentIds = Object.keys(message.mimeTree.attachmentMap || {}).map(key => message.mimeTree.attachmentMap[key]);
|
|
|
|
|
|
|
|
if (!attachmentIds.length) {
|
|
|
|
let entry = {
|
|
|
|
command: 'EXISTS',
|
|
|
|
uid: message.uid,
|
|
|
|
message: message._id,
|
|
|
|
unseen: message.unseen
|
|
|
|
};
|
|
|
|
if (junk) {
|
|
|
|
entry.junk = junk;
|
|
|
|
}
|
|
|
|
return server.notifier.addEntries(targetData, entry, processNext);
|
|
|
|
}
|
|
|
|
|
|
|
|
messageHandler.attachmentStorage.updateMany(attachmentIds, 1, message.magic, err => {
|
|
|
|
if (err) {
|
|
|
|
// should we care about this error?
|
|
|
|
}
|
|
|
|
let entry = {
|
|
|
|
command: 'EXISTS',
|
|
|
|
uid: message.uid,
|
|
|
|
message: message._id,
|
|
|
|
unseen: message.unseen
|
|
|
|
};
|
|
|
|
if (junk) {
|
|
|
|
entry.junk = junk;
|
|
|
|
}
|
|
|
|
server.notifier.addEntries(targetData, entry, processNext);
|
|
|
|
});
|
|
|
|
});
|
2017-07-16 00:08:33 +08:00
|
|
|
}
|
2017-12-10 07:19:50 +08:00
|
|
|
);
|
2017-07-16 00:08:33 +08:00
|
|
|
});
|
2017-12-10 07:19:50 +08:00
|
|
|
};
|
|
|
|
processNext();
|
|
|
|
}
|
|
|
|
);
|
|
|
|
}
|
|
|
|
);
|
2017-07-16 00:08:33 +08:00
|
|
|
};
|