2017-07-21 02:33:41 +08:00
|
|
|
'use strict';
|
|
|
|
|
|
|
|
const ObjectID = require('mongodb').ObjectID;
|
|
|
|
const ImapNotifier = require('./imap-notifier');
|
|
|
|
|
|
|
|
class MailboxHandler {
|
|
|
|
constructor(options) {
|
|
|
|
this.database = options.database;
|
|
|
|
this.users = options.users || options.database;
|
|
|
|
this.redis = options.redis;
|
2018-10-18 15:37:32 +08:00
|
|
|
|
|
|
|
this.loggelf = options.loggelf || (() => false);
|
|
|
|
|
2017-07-21 02:33:41 +08:00
|
|
|
this.notifier =
|
|
|
|
options.notifier ||
|
|
|
|
new ImapNotifier({
|
|
|
|
database: options.database,
|
|
|
|
redis: this.redis,
|
|
|
|
pushOnly: true
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
|
|
|
create(user, path, opts, callback) {
|
2017-12-10 07:19:50 +08:00
|
|
|
this.database.collection('mailboxes').findOne(
|
|
|
|
{
|
|
|
|
user,
|
|
|
|
path
|
|
|
|
},
|
|
|
|
(err, mailboxData) => {
|
2017-07-21 02:33:41 +08:00
|
|
|
if (err) {
|
|
|
|
return callback(err);
|
|
|
|
}
|
2017-12-10 07:19:50 +08:00
|
|
|
if (mailboxData) {
|
|
|
|
return callback(null, 'ALREADYEXISTS');
|
2017-07-21 02:33:41 +08:00
|
|
|
}
|
|
|
|
|
2017-12-10 07:19:50 +08:00
|
|
|
this.users.collection('users').findOne(
|
|
|
|
{
|
|
|
|
_id: user
|
|
|
|
},
|
|
|
|
{
|
2018-08-15 04:45:45 +08:00
|
|
|
projection: {
|
2017-12-10 07:19:50 +08:00
|
|
|
retention: true
|
|
|
|
}
|
|
|
|
},
|
|
|
|
(err, userData) => {
|
|
|
|
if (err) {
|
|
|
|
return callback(err);
|
|
|
|
}
|
2017-07-21 02:33:41 +08:00
|
|
|
|
2017-12-10 07:19:50 +08:00
|
|
|
if (!userData) {
|
|
|
|
return callback(new Error('User not found'));
|
2017-07-30 23:07:35 +08:00
|
|
|
}
|
2017-12-10 07:19:50 +08:00
|
|
|
|
|
|
|
mailboxData = {
|
|
|
|
_id: new ObjectID(),
|
|
|
|
user,
|
|
|
|
path,
|
|
|
|
uidValidity: Math.floor(Date.now() / 1000),
|
|
|
|
uidNext: 1,
|
|
|
|
modifyIndex: 0,
|
|
|
|
subscribed: true,
|
|
|
|
flags: [],
|
|
|
|
retention: userData.retention
|
|
|
|
};
|
|
|
|
|
|
|
|
Object.keys(opts || {}).forEach(key => {
|
|
|
|
if (!['_id', 'user', 'path'].includes(key)) {
|
|
|
|
mailboxData[key] = opts[key];
|
|
|
|
}
|
|
|
|
});
|
|
|
|
|
2018-10-25 00:21:10 +08:00
|
|
|
this.database.collection('mailboxes').insertOne(mailboxData, { w: 'majority' }, (err, r) => {
|
2017-12-10 07:19:50 +08:00
|
|
|
if (err) {
|
|
|
|
if (err.code === 11000) {
|
|
|
|
return callback(null, 'ALREADYEXISTS');
|
|
|
|
}
|
|
|
|
return callback(err);
|
|
|
|
}
|
|
|
|
return this.notifier.addEntries(
|
|
|
|
mailboxData,
|
|
|
|
{
|
|
|
|
command: 'CREATE',
|
|
|
|
mailbox: r.insertedId,
|
|
|
|
path
|
|
|
|
},
|
|
|
|
() => {
|
|
|
|
this.notifier.fire(user);
|
|
|
|
return callback(null, true, mailboxData._id);
|
|
|
|
}
|
|
|
|
);
|
|
|
|
});
|
2017-07-21 02:33:41 +08:00
|
|
|
}
|
2017-12-10 07:19:50 +08:00
|
|
|
);
|
|
|
|
}
|
|
|
|
);
|
2017-07-21 02:33:41 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
rename(user, mailbox, newname, opts, callback) {
|
2017-12-10 07:19:50 +08:00
|
|
|
this.database.collection('mailboxes').findOne(
|
|
|
|
{
|
|
|
|
_id: mailbox,
|
|
|
|
user
|
|
|
|
},
|
|
|
|
(err, mailboxData) => {
|
2017-07-21 02:33:41 +08:00
|
|
|
if (err) {
|
|
|
|
return callback(err);
|
|
|
|
}
|
2017-12-10 07:19:50 +08:00
|
|
|
if (!mailboxData) {
|
|
|
|
return callback(null, 'NONEXISTENT');
|
2017-07-21 02:33:41 +08:00
|
|
|
}
|
2017-12-10 07:19:50 +08:00
|
|
|
if (mailboxData.path === 'INBOX') {
|
|
|
|
return callback(null, 'CANNOT');
|
|
|
|
}
|
|
|
|
this.database.collection('mailboxes').findOne(
|
|
|
|
{
|
|
|
|
user: mailboxData.user,
|
|
|
|
path: newname
|
|
|
|
},
|
|
|
|
(err, existing) => {
|
|
|
|
if (err) {
|
|
|
|
return callback(err);
|
|
|
|
}
|
|
|
|
if (existing) {
|
|
|
|
return callback(null, 'ALREADYEXISTS');
|
|
|
|
}
|
2017-07-21 02:33:41 +08:00
|
|
|
|
2017-12-10 07:19:50 +08:00
|
|
|
let $set = { path: newname };
|
2017-07-21 02:33:41 +08:00
|
|
|
|
2017-12-10 07:19:50 +08:00
|
|
|
Object.keys(opts || {}).forEach(key => {
|
|
|
|
if (!['_id', 'user', 'path'].includes(key)) {
|
|
|
|
$set[key] = opts[key];
|
|
|
|
}
|
|
|
|
});
|
2017-07-21 02:33:41 +08:00
|
|
|
|
2017-12-10 07:19:50 +08:00
|
|
|
this.database.collection('mailboxes').findOneAndUpdate(
|
|
|
|
{
|
|
|
|
_id: mailbox
|
|
|
|
},
|
|
|
|
{
|
|
|
|
$set
|
|
|
|
},
|
|
|
|
{},
|
|
|
|
(err, item) => {
|
|
|
|
if (err) {
|
|
|
|
return callback(err);
|
|
|
|
}
|
2017-07-21 02:33:41 +08:00
|
|
|
|
2017-12-10 07:19:50 +08:00
|
|
|
if (!item || !item.value) {
|
|
|
|
// was not able to acquire a lock
|
|
|
|
return callback(null, 'NONEXISTENT');
|
|
|
|
}
|
|
|
|
this.notifier.addEntries(
|
|
|
|
mailboxData,
|
|
|
|
{
|
|
|
|
command: 'RENAME',
|
|
|
|
path: newname
|
|
|
|
},
|
|
|
|
() => {
|
|
|
|
this.notifier.fire(mailboxData.user);
|
2018-11-28 16:58:45 +08:00
|
|
|
return callback(null, true, mailbox);
|
2017-12-10 07:19:50 +08:00
|
|
|
}
|
|
|
|
);
|
|
|
|
}
|
|
|
|
);
|
2017-07-21 02:33:41 +08:00
|
|
|
}
|
2017-12-10 07:19:50 +08:00
|
|
|
);
|
|
|
|
}
|
|
|
|
);
|
2017-07-21 02:33:41 +08:00
|
|
|
}
|
|
|
|
|
2017-11-17 19:37:53 +08:00
|
|
|
/**
|
|
|
|
* Deletes a mailbox. Does not immediatelly release quota as the messages get deleted after a while
|
|
|
|
*/
|
2017-07-21 02:33:41 +08:00
|
|
|
del(user, mailbox, callback) {
|
2017-12-10 07:19:50 +08:00
|
|
|
this.database.collection('mailboxes').findOne(
|
|
|
|
{
|
|
|
|
_id: mailbox,
|
|
|
|
user
|
|
|
|
},
|
|
|
|
(err, mailboxData) => {
|
2017-07-21 02:33:41 +08:00
|
|
|
if (err) {
|
|
|
|
return callback(err);
|
|
|
|
}
|
2017-12-10 07:19:50 +08:00
|
|
|
if (!mailboxData) {
|
|
|
|
return callback(null, 'NONEXISTENT');
|
|
|
|
}
|
|
|
|
if (mailboxData.specialUse || mailboxData.path === 'INBOX') {
|
|
|
|
return callback(null, 'CANNOT');
|
|
|
|
}
|
2017-07-21 02:33:41 +08:00
|
|
|
|
2017-12-10 07:19:50 +08:00
|
|
|
this.database.collection('mailboxes').deleteOne(
|
2017-07-21 02:33:41 +08:00
|
|
|
{
|
2017-12-10 07:19:50 +08:00
|
|
|
_id: mailbox
|
2017-07-21 02:33:41 +08:00
|
|
|
},
|
2018-10-25 00:21:10 +08:00
|
|
|
{ w: 'majority' },
|
2017-12-10 07:19:50 +08:00
|
|
|
err => {
|
|
|
|
if (err) {
|
|
|
|
return callback(err);
|
|
|
|
}
|
2017-11-17 19:37:53 +08:00
|
|
|
|
2019-03-20 21:00:57 +08:00
|
|
|
// delete matching filters as well
|
2019-03-20 21:12:11 +08:00
|
|
|
this.database.collection('filters').deleteMany(
|
2017-12-10 07:19:50 +08:00
|
|
|
{
|
2019-03-20 21:00:57 +08:00
|
|
|
user,
|
|
|
|
'action.mailbox': mailbox
|
2017-12-10 07:19:50 +08:00
|
|
|
},
|
2019-03-20 21:00:57 +08:00
|
|
|
err => {
|
|
|
|
if (err) {
|
|
|
|
this.loggelf({
|
|
|
|
user,
|
2017-12-10 07:19:50 +08:00
|
|
|
mailbox,
|
2019-03-20 21:00:57 +08:00
|
|
|
action: 'delete_filter',
|
|
|
|
error: err.message
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
|
|
|
// send information about deleted mailbox straightly to connected clients
|
|
|
|
this.notifier.fire(mailboxData.user, {
|
|
|
|
command: 'DROP',
|
|
|
|
mailbox
|
|
|
|
});
|
|
|
|
|
|
|
|
this.notifier.addEntries(
|
|
|
|
mailboxData,
|
2017-12-10 07:19:50 +08:00
|
|
|
{
|
2019-03-20 21:00:57 +08:00
|
|
|
command: 'DELETE',
|
|
|
|
mailbox
|
2017-12-10 07:19:50 +08:00
|
|
|
},
|
2019-03-20 21:00:57 +08:00
|
|
|
() => {
|
|
|
|
this.database.collection('messages').updateMany(
|
|
|
|
{
|
|
|
|
mailbox,
|
|
|
|
uid: {
|
|
|
|
$gt: 0,
|
|
|
|
$lt: mailboxData.uidNext + 100
|
|
|
|
}
|
|
|
|
},
|
|
|
|
{
|
|
|
|
$set: {
|
|
|
|
exp: true,
|
|
|
|
// make sure the messages are in top of the expire queue
|
|
|
|
rdate: Date.now() - 24 * 3600 * 1000
|
|
|
|
}
|
|
|
|
},
|
|
|
|
{
|
|
|
|
multi: true,
|
|
|
|
w: 1
|
|
|
|
},
|
|
|
|
err => {
|
|
|
|
if (err) {
|
|
|
|
return callback(err);
|
|
|
|
}
|
2017-12-10 07:19:50 +08:00
|
|
|
|
2019-03-20 21:00:57 +08:00
|
|
|
let done = () => {
|
|
|
|
this.notifier.fire(mailboxData.user);
|
|
|
|
callback(null, true, mailbox);
|
|
|
|
};
|
2017-12-10 07:19:50 +08:00
|
|
|
|
2019-03-20 21:00:57 +08:00
|
|
|
return done();
|
|
|
|
}
|
|
|
|
);
|
2017-12-10 07:19:50 +08:00
|
|
|
}
|
|
|
|
);
|
|
|
|
}
|
|
|
|
);
|
2017-07-21 02:33:41 +08:00
|
|
|
}
|
|
|
|
);
|
2017-12-10 07:19:50 +08:00
|
|
|
}
|
|
|
|
);
|
2017-07-21 02:33:41 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
update(user, mailbox, updates, callback) {
|
|
|
|
if (!updates) {
|
|
|
|
return callback(null, false);
|
|
|
|
}
|
|
|
|
|
2017-12-10 07:19:50 +08:00
|
|
|
this.database.collection('mailboxes').findOne(
|
|
|
|
{
|
2017-07-21 02:33:41 +08:00
|
|
|
_id: mailbox
|
2017-12-10 07:19:50 +08:00
|
|
|
},
|
|
|
|
(err, mailboxData) => {
|
2017-07-21 02:33:41 +08:00
|
|
|
if (err) {
|
|
|
|
return callback(err);
|
|
|
|
}
|
2017-12-10 07:19:50 +08:00
|
|
|
if (!mailboxData) {
|
2017-07-21 02:33:41 +08:00
|
|
|
return callback(null, 'NONEXISTENT');
|
|
|
|
}
|
2017-12-10 07:19:50 +08:00
|
|
|
if (updates.path !== mailboxData.path) {
|
|
|
|
return this.rename(user, mailbox, updates.path, updates, callback);
|
|
|
|
}
|
2017-07-21 02:33:41 +08:00
|
|
|
|
2017-12-10 07:19:50 +08:00
|
|
|
let $set = {};
|
2018-06-26 19:46:03 +08:00
|
|
|
let hasChanges = false;
|
2017-12-10 07:19:50 +08:00
|
|
|
|
|
|
|
Object.keys(updates || {}).forEach(key => {
|
|
|
|
if (!['_id', 'user', 'path'].includes(key)) {
|
|
|
|
$set[key] = updates[key];
|
2018-06-26 19:46:03 +08:00
|
|
|
hasChanges = true;
|
2017-12-10 07:19:50 +08:00
|
|
|
}
|
|
|
|
});
|
|
|
|
|
2018-06-26 19:46:03 +08:00
|
|
|
if (!hasChanges) {
|
|
|
|
return callback(null, true);
|
|
|
|
}
|
|
|
|
|
2017-12-10 07:19:50 +08:00
|
|
|
this.database.collection('mailboxes').findOneAndUpdate(
|
|
|
|
{
|
|
|
|
_id: mailbox
|
|
|
|
},
|
|
|
|
{
|
|
|
|
$set
|
|
|
|
},
|
|
|
|
{},
|
|
|
|
(err, item) => {
|
|
|
|
if (err) {
|
|
|
|
return callback(err);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!item || !item.value) {
|
|
|
|
return callback(null, 'NONEXISTENT');
|
|
|
|
}
|
|
|
|
|
|
|
|
return callback(null, true);
|
|
|
|
}
|
|
|
|
);
|
|
|
|
}
|
|
|
|
);
|
2017-07-21 02:33:41 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
module.exports = MailboxHandler;
|