wildduck/imap.js

428 lines
14 KiB
JavaScript
Raw Normal View History

2017-03-06 22:13:40 +08:00
'use strict';
const log = require('npmlog');
2017-07-16 19:37:33 +08:00
const config = require('wild-config');
2017-03-06 22:13:40 +08:00
const IMAPServerModule = require('./imap-core');
const IMAPServer = IMAPServerModule.IMAPServer;
const ImapNotifier = require('./lib/imap-notifier');
2017-03-06 22:13:40 +08:00
const Indexer = require('./imap-core/lib/indexer/indexer');
const MessageHandler = require('./lib/message-handler');
2017-04-21 01:10:03 +08:00
const UserHandler = require('./lib/user-handler');
2017-07-21 02:33:41 +08:00
const MailboxHandler = require('./lib/mailbox-handler');
const db = require('./lib/db');
const consts = require('./lib/consts');
2017-10-03 16:18:23 +08:00
const RedFour = require('ioredfour');
const packageData = require('./package.json');
const yaml = require('js-yaml');
const fs = require('fs');
const certs = require('./lib/certs');
2017-09-26 14:58:37 +08:00
const setupIndexes = yaml.safeLoad(fs.readFileSync(__dirname + '/indexes.yaml', 'utf8'));
2017-03-06 22:13:40 +08:00
const onFetch = require('./lib/handlers/on-fetch');
const onAuth = require('./lib/handlers/on-auth');
const onList = require('./lib/handlers/on-list');
const onLsub = require('./lib/handlers/on-lsub');
const onSubscribe = require('./lib/handlers/on-subscribe');
const onUnsubscribe = require('./lib/handlers/on-unsubscribe');
const onCreate = require('./lib/handlers/on-create');
const onRename = require('./lib/handlers/on-rename');
const onDelete = require('./lib/handlers/on-delete');
const onOpen = require('./lib/handlers/on-open');
const onStatus = require('./lib/handlers/on-status');
const onAppend = require('./lib/handlers/on-append');
const onStore = require('./lib/handlers/on-store');
const onExpunge = require('./lib/handlers/on-expunge');
const onCopy = require('./lib/handlers/on-copy');
const onMove = require('./lib/handlers/on-move');
const onSearch = require('./lib/handlers/on-search');
const onGetQuotaRoot = require('./lib/handlers/on-get-quota-root');
const onGetQuota = require('./lib/handlers/on-get-quota');
// Setup server
const serverOptions = {
secure: config.imap.secure,
2017-09-11 03:53:12 +08:00
disableSTARTTLS: config.imap.disableSTARTTLS,
ignoreSTARTTLS: config.imap.ignoreSTARTTLS,
2017-03-06 22:13:40 +08:00
id: {
2017-07-27 16:47:32 +08:00
name: config.imap.name || 'Wild Duck IMAP Server',
version: config.imap.version || packageData.version,
vendor: config.imap.vendor || 'Kreata'
},
logger: {
info(...args) {
args.shift();
log.info('IMAP', ...args);
2017-04-09 17:33:10 +08:00
},
debug(...args) {
args.shift();
log.silly('IMAP', ...args);
},
error(...args) {
args.shift();
log.error('IMAP', ...args);
2017-03-06 22:13:40 +08:00
}
},
2017-03-06 22:13:40 +08:00
maxMessage: config.imap.maxMB * 1024 * 1024,
maxStorage: config.maxStorage * 1024 * 1024
};
2017-03-06 22:13:40 +08:00
certs.loadTLSOptions(serverOptions, 'imap');
2017-03-21 05:19:25 +08:00
const server = new IMAPServer(serverOptions);
2017-03-06 22:13:40 +08:00
certs.registerReload(server, 'imap');
let messageHandler;
let userHandler;
2017-07-21 02:33:41 +08:00
let mailboxHandler;
let gcTimeout;
let gcLock;
function clearExpiredMessages() {
clearTimeout(gcTimeout);
let startTime = Date.now();
// First, acquire the lock. This prevents multiple connected clients for deleting the same messages
gcLock.acquireLock('gc_expired', Math.round(consts.GC_INTERVAL * 1.2) /* Lock expires if not released */, (err, lock) => {
if (err) {
server.logger.error(
{
tnx: 'gc',
err
},
'Failed to acquire lock error=%s',
err.message
);
gcTimeout = setTimeout(clearExpiredMessages, consts.GC_INTERVAL);
gcTimeout.unref();
return;
} else if (!lock.success) {
gcTimeout = setTimeout(clearExpiredMessages, consts.GC_INTERVAL);
gcTimeout.unref();
return;
}
let done = () => {
gcLock.releaseLock(lock, err => {
if (err) {
server.logger.error(
{
tnx: 'gc',
err
},
'Failed to release lock error=%s',
err.message
);
}
gcTimeout = setTimeout(clearExpiredMessages, consts.GC_INTERVAL);
gcTimeout.unref();
});
};
2017-06-06 14:19:35 +08:00
if (config.imap.disableRetention) {
// delete all attachments that do not have any active links to message objects
return messageHandler.attachmentStorage.deleteOrphaned(() => done(null, true));
2017-06-06 14:19:35 +08:00
}
2017-07-17 21:32:31 +08:00
// find and delete all messages that are expired
2017-08-11 03:20:21 +08:00
// NB! scattered query, searches over all mailboxes and thus over all shards
let cursor = db.database
.collection('messages')
.find({
exp: true,
rdate: {
$lte: Date.now()
}
})
.project({
_id: true,
mailbox: true,
uid: true,
size: true,
2017-08-07 02:25:10 +08:00
'mimeTree.attachmentMap': true,
2017-07-20 21:10:36 +08:00
magic: true,
unseen: true
});
let deleted = 0;
let clear = () =>
cursor.close(() => {
// delete all attachments that do not have any active links to message objects
messageHandler.attachmentStorage.deleteOrphaned(() => {
2017-07-31 15:59:18 +08:00
if (deleted) {
server.logger.debug(
{
tnx: 'gc'
},
'Deleted %s messages',
deleted
);
}
done(null, true);
});
});
let processNext = () => {
if (Date.now() - startTime > consts.GC_INTERVAL * 0.8) {
2017-07-17 21:32:31 +08:00
// deleting expired messages has taken too long time, cancel
return clear();
}
cursor.next((err, message) => {
if (err) {
return done(err);
}
if (!message) {
return clear();
}
server.logger.info(
{
tnx: 'gc',
err
},
'Deleting expired message id=%s',
message._id
);
gcTimeout = setTimeout(clearExpiredMessages, consts.GC_INTERVAL);
messageHandler.del(
{
message,
skipAttachments: true
},
err => {
if (err) {
return cursor.close(() => done(err));
}
deleted++;
if (consts.GC_DELAY_DELETE) {
setTimeout(processNext, consts.GC_DELAY_DELETE);
2017-06-06 14:19:35 +08:00
} else {
setImmediate(processNext);
}
}
);
});
};
processNext();
});
}
module.exports = done => {
2017-04-13 16:35:39 +08:00
if (!config.imap.enabled) {
return setImmediate(() => done(null, false));
}
2017-07-12 02:38:23 +08:00
gcLock = new RedFour({
2017-10-03 16:18:23 +08:00
redis: db.redis,
2017-07-12 02:38:23 +08:00
namespace: 'wildduck'
});
gcTimeout = setTimeout(clearExpiredMessages, consts.GC_INTERVAL);
2017-07-12 02:38:23 +08:00
gcTimeout.unref();
let start = () => {
server.indexer = new Indexer({
database: db.database
});
2017-03-06 22:13:40 +08:00
// setup notification system for updates
server.notifier = new ImapNotifier({
2017-07-17 21:32:31 +08:00
database: db.database,
redis: db.redis
});
2017-08-07 02:25:10 +08:00
messageHandler = new MessageHandler({
database: db.database,
redis: db.redis,
2017-08-07 02:25:10 +08:00
gridfs: db.gridfs,
attachments: config.attachments
2017-08-07 02:25:10 +08:00
});
userHandler = new UserHandler({
database: db.database,
users: db.users,
2017-08-08 19:35:18 +08:00
redis: db.redis,
authlogExpireDays: config.log.authlogExpireDays
2017-08-07 02:25:10 +08:00
});
mailboxHandler = new MailboxHandler({
database: db.database,
users: db.users,
redis: db.redis,
notifier: server.notifier
});
2017-07-21 02:33:41 +08:00
let started = false;
server.on('error', err => {
if (!started) {
2017-03-06 22:13:40 +08:00
started = true;
return done(err);
}
server.logger.error(
{
err
},
2017-04-08 02:29:14 +08:00
err
);
});
2017-03-06 22:13:40 +08:00
// start listening
server.listen(config.imap.port, config.imap.host, () => {
if (started) {
return server.close();
2017-03-06 22:13:40 +08:00
}
started = true;
done(null, server);
});
// setup command handlers for the server instance
2017-08-07 02:25:10 +08:00
server.onFetch = onFetch(server, messageHandler);
server.onAuth = onAuth(server, userHandler);
server.onList = onList(server);
server.onLsub = onLsub(server);
server.onSubscribe = onSubscribe(server);
server.onUnsubscribe = onUnsubscribe(server);
2017-07-21 02:33:41 +08:00
server.onCreate = onCreate(server, mailboxHandler);
server.onRename = onRename(server, mailboxHandler);
server.onDelete = onDelete(server, mailboxHandler);
server.onOpen = onOpen(server);
server.onStatus = onStatus(server);
server.onAppend = onAppend(server, messageHandler);
server.onStore = onStore(server);
2017-08-07 02:25:10 +08:00
server.onExpunge = onExpunge(server, messageHandler);
server.onCopy = onCopy(server, messageHandler);
server.onMove = onMove(server, messageHandler);
server.onSearch = onSearch(server);
server.onGetQuotaRoot = onGetQuotaRoot(server);
server.onGetQuota = onGetQuota(server);
};
2017-09-26 14:58:37 +08:00
let collections = setupIndexes.collections;
let collectionpos = 0;
let ensureCollections = next => {
if (collectionpos >= collections.length) {
server.logger.info(
{
tnx: 'mongo'
},
'Setup %s collections',
collections.length
);
return next();
}
let collection = collections[collectionpos++];
db[collection.type || 'database'].createCollection(collection.collection, collection.options, err => {
if (err) {
server.logger.error(
{
err,
tnx: 'mongo'
},
'Failed creating collection %s %s. %s',
collectionpos,
JSON.stringify(collection.collection),
err.message
);
}
ensureCollections(next);
});
};
let indexes = setupIndexes.indexes;
let indexpos = 0;
2017-07-12 02:38:23 +08:00
let ensureIndexes = next => {
2017-09-26 14:58:37 +08:00
if (indexpos >= indexes.length) {
server.logger.info(
{
tnx: 'mongo'
},
'Setup indexes for %s collections',
2017-09-26 14:58:37 +08:00
indexes.length
);
2017-07-12 02:38:23 +08:00
return next();
}
2017-09-26 14:58:37 +08:00
let index = indexes[indexpos++];
db[index.type || 'database'].collection(index.collection).createIndexes([index.index], (err, r) => {
2017-07-12 02:38:23 +08:00
if (err) {
server.logger.error(
{
err,
tnx: 'mongo'
},
'Failed creating index %s %s. %s',
2017-07-12 02:38:23 +08:00
indexpos,
2017-07-24 21:44:08 +08:00
JSON.stringify(index.collection + '.' + index.index.name),
2017-07-12 02:38:23 +08:00
err.message
);
} else if (r.numIndexesAfter !== r.numIndexesBefore) {
server.logger.debug(
{
tnx: 'mongo'
},
'Created index %s %s',
indexpos,
2017-07-24 21:44:08 +08:00
JSON.stringify(index.collection + '.' + index.index.name)
2017-07-12 02:38:23 +08:00
);
} else {
server.logger.debug(
{
tnx: 'mongo'
},
'Skipped index %s %s: %s',
indexpos,
2017-07-24 21:44:08 +08:00
JSON.stringify(index.collection + '.' + index.index.name),
2017-07-12 02:38:23 +08:00
r.note || 'No index added'
);
}
2017-07-12 02:38:23 +08:00
ensureIndexes(next);
});
};
gcLock.acquireLock('db_indexes', 1 * 60 * 1000, (err, lock) => {
2017-07-12 02:38:23 +08:00
if (err) {
server.logger.error(
{
tnx: 'gc',
err
},
'Failed to acquire lock error=%s',
err.message
);
return start();
} else if (!lock.success) {
return start();
}
2017-07-12 02:38:23 +08:00
2017-09-26 14:58:37 +08:00
ensureCollections(() => {
ensureIndexes(() => {
// Do not release the indexing lock immediatelly
setTimeout(() => {
gcLock.releaseLock(lock, err => {
if (err) {
server.logger.error(
{
tnx: 'gc',
err
},
'Failed to release lock error=%s',
err.message
);
}
});
}, 60 * 1000);
return start();
});
2017-07-12 02:38:23 +08:00
});
});
2017-03-06 22:13:40 +08:00
};