2017-03-06 05:45:50 +08:00
|
|
|
'use strict';
|
|
|
|
|
|
|
|
let imapHandler = require('./handler/imap-handler');
|
|
|
|
|
|
|
|
const MAX_MESSAGE_SIZE = 1 * 1024 * 1024;
|
|
|
|
|
|
|
|
let commands = new Map([
|
|
|
|
/*eslint-disable global-require*/
|
|
|
|
// require must normally be on top of the module
|
|
|
|
['NOOP', require('./commands/noop')],
|
|
|
|
['CAPABILITY', require('./commands/capability')],
|
|
|
|
['LOGOUT', require('./commands/logout')],
|
|
|
|
['ID', require('./commands/id')],
|
|
|
|
['STARTTLS', require('./commands/starttls')],
|
|
|
|
['LOGIN', require('./commands/login')],
|
|
|
|
['AUTHENTICATE PLAIN', require('./commands/authenticate-plain')],
|
|
|
|
['NAMESPACE', require('./commands/namespace')],
|
|
|
|
['LIST', require('./commands/list')],
|
|
|
|
['LSUB', require('./commands/lsub')],
|
|
|
|
['SUBSCRIBE', require('./commands/subscribe')],
|
|
|
|
['UNSUBSCRIBE', require('./commands/unsubscribe')],
|
|
|
|
['CREATE', require('./commands/create')],
|
|
|
|
['DELETE', require('./commands/delete')],
|
|
|
|
['RENAME', require('./commands/rename')],
|
|
|
|
['SELECT', require('./commands/select')],
|
|
|
|
['EXAMINE', require('./commands/select')],
|
|
|
|
['IDLE', require('./commands/idle')],
|
|
|
|
['CHECK', require('./commands/check')],
|
|
|
|
['STATUS', require('./commands/status')],
|
|
|
|
['APPEND', require('./commands/append')],
|
|
|
|
['STORE', require('./commands/store')],
|
|
|
|
['UID STORE', require('./commands/uid-store')],
|
|
|
|
['EXPUNGE', require('./commands/expunge')],
|
|
|
|
['UID EXPUNGE', require('./commands/uid-expunge')],
|
|
|
|
['CLOSE', require('./commands/close')],
|
|
|
|
['UNSELECT', require('./commands/unselect')],
|
|
|
|
['COPY', require('./commands/copy')],
|
|
|
|
['UID COPY', require('./commands/copy')],
|
|
|
|
['FETCH', require('./commands/fetch')],
|
|
|
|
['UID FETCH', require('./commands/fetch')],
|
|
|
|
['SEARCH', require('./commands/search')],
|
|
|
|
['UID SEARCH', require('./commands/search')],
|
|
|
|
['ENABLE', require('./commands/enable')]
|
|
|
|
/*eslint-enable global-require*/
|
|
|
|
]);
|
|
|
|
|
|
|
|
class IMAPCommand {
|
|
|
|
|
|
|
|
constructor(connection) {
|
|
|
|
this.connection = connection;
|
|
|
|
this.payload = '';
|
|
|
|
this.first = true;
|
|
|
|
}
|
|
|
|
|
|
|
|
append(command, callback) {
|
|
|
|
let chunks = [];
|
|
|
|
let chunklen = 0;
|
|
|
|
|
|
|
|
this.payload += command.value;
|
|
|
|
|
|
|
|
if (this.first) {
|
|
|
|
// fetch tag and command name
|
|
|
|
this.first = false;
|
|
|
|
|
|
|
|
// only check payload if it is a regular command, not input for something else
|
|
|
|
if (typeof this.connection._nextHandler !== 'function') {
|
|
|
|
let match = /^([^\s]+)(?:\s+((?:AUTHENTICATE |UID )?[^\s]+)|$)/i.exec(command.value) || [];
|
|
|
|
this.tag = match[1];
|
|
|
|
this.command = (match[2] || '').trim().toUpperCase();
|
|
|
|
|
|
|
|
if (!this.command || !this.tag) {
|
|
|
|
this.connection.send('* BAD Invalid tag');
|
|
|
|
return callback(new Error('Invalid tag'));
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!commands.has(this.command)) {
|
|
|
|
this.connection.send(this.tag + ' BAD Unknown command: ' + this.command);
|
|
|
|
return callback(new Error('Unknown command'));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (command.literal) {
|
|
|
|
if (
|
|
|
|
// Allow large literals for selected commands only
|
|
|
|
(['APPEND'].indexOf(this.command) < 0 && command.expecting > 1024) ||
|
|
|
|
// Deny all literals bigger than maxMessage
|
|
|
|
command.expecting > Math.max(Number(this.connection._server.options.maxMessage) || 0, MAX_MESSAGE_SIZE)) {
|
|
|
|
|
|
|
|
this.connection._server.logger.debug('[%s] C:', this.connection.id, this.payload);
|
|
|
|
this.connection.send(this.tag + ' NO Literal too big');
|
|
|
|
return callback(new Error('Literal too big'));
|
|
|
|
}
|
|
|
|
|
|
|
|
// Accept literal input
|
|
|
|
this.connection.send('+ Go ahead');
|
|
|
|
|
|
|
|
// currently the stream is buffered into a large string and thats it.
|
|
|
|
// in the future we might consider some kind of actual stream usage
|
|
|
|
command.literal.on('data', chunk => {
|
|
|
|
chunks.push(chunk);
|
|
|
|
chunklen += chunk.length;
|
|
|
|
});
|
|
|
|
|
|
|
|
command.literal.on('end', () => {
|
|
|
|
this.payload += '\r\n' + Buffer.concat(chunks, chunklen).toString('binary');
|
|
|
|
command.readyCallback(); // call this once stream is fully processed and ready to accept next data
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
|
|
|
callback();
|
|
|
|
}
|
|
|
|
|
|
|
|
end(command, callback) {
|
|
|
|
let callbackSent = false;
|
|
|
|
let next = err => {
|
|
|
|
if (!callbackSent) {
|
|
|
|
callbackSent = true;
|
|
|
|
return callback(err);
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
this.append(command, err => {
|
|
|
|
if (err) {
|
|
|
|
return next(err);
|
|
|
|
}
|
|
|
|
|
2017-03-11 21:59:38 +08:00
|
|
|
|
2017-03-06 05:45:50 +08:00
|
|
|
|
|
|
|
// check if the payload needs to be directod to a preset handler
|
|
|
|
if (typeof this.connection._nextHandler === 'function') {
|
2017-03-11 21:59:38 +08:00
|
|
|
this.connection._server.logger.debug('[%s] C: <%s bytes of data>', this.connection.id, this.payload && this.payload.length || 0);
|
2017-03-06 05:45:50 +08:00
|
|
|
return this.connection._nextHandler(this.payload, next);
|
|
|
|
}
|
|
|
|
|
|
|
|
try {
|
|
|
|
this.parsed = imapHandler.parser(this.payload);
|
|
|
|
} catch (E) {
|
2017-03-11 21:59:38 +08:00
|
|
|
this.connection._server.logger.debug('[%s] C:', this.connection.id, this.payload);
|
2017-03-06 05:45:50 +08:00
|
|
|
this.connection.send(this.tag + ' BAD ' + E.message);
|
|
|
|
return next();
|
|
|
|
}
|
|
|
|
|
|
|
|
let handler = commands.get(this.command);
|
|
|
|
|
2017-03-11 21:59:38 +08:00
|
|
|
if (/^(AUTHENTICATE|LOGIN)/.test(this.command) && Array.isArray(this.parsed.attributes)) {
|
|
|
|
this.parsed.attributes.forEach(attr => {
|
|
|
|
if (attr && typeof attr === 'object' && attr.value) {
|
|
|
|
attr.sensitive = true;
|
|
|
|
}
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
|
|
|
this.connection._server.logger.debug('[%s] C:', this.connection.id, imapHandler.compiler(this.parsed, false, true));
|
|
|
|
|
2017-03-06 05:45:50 +08:00
|
|
|
this.validateCommand(this.parsed, handler, err => {
|
|
|
|
if (err) {
|
|
|
|
this.connection.send(this.tag + ' ' + (err.response || 'BAD') + ' ' + err.message);
|
|
|
|
return next(err);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (typeof handler.handler === 'function') {
|
|
|
|
handler.handler.call(this.connection, this.parsed, (err, response) => {
|
|
|
|
if (err) {
|
|
|
|
this.connection.send(this.tag + ' ' + (err.response || 'BAD') + ' ' + err.message);
|
|
|
|
return next(err);
|
|
|
|
}
|
|
|
|
|
|
|
|
// send EXPUNGE, EXISTS etc queued notices
|
|
|
|
this.sendNotifications(handler, () => {
|
|
|
|
|
|
|
|
// send command ready response
|
|
|
|
this.connection.writeStream.write({
|
|
|
|
tag: this.tag,
|
|
|
|
command: response.response,
|
|
|
|
attributes: [].concat(response.code ? {
|
|
|
|
type: 'SECTION',
|
|
|
|
section: [{
|
|
|
|
type: 'TEXT',
|
|
|
|
value: response.code
|
|
|
|
}]
|
|
|
|
} : []).concat({
|
|
|
|
type: 'TEXT',
|
|
|
|
value: response.message || this.command + ' completed'
|
|
|
|
})
|
|
|
|
});
|
|
|
|
|
|
|
|
next();
|
|
|
|
});
|
|
|
|
}, next);
|
|
|
|
} else {
|
|
|
|
this.connection.send(this.tag + ' NO Not implemented: ' + this.command);
|
|
|
|
return next();
|
|
|
|
}
|
|
|
|
});
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
|
|
|
sendNotifications(handler, callback) {
|
|
|
|
if (this.connection.state !== 'Selected' || !!handler.disableNotifications) {
|
|
|
|
// nothing to advertise if not in Selected state
|
|
|
|
return callback();
|
|
|
|
}
|
|
|
|
|
|
|
|
this.connection.emitNotifications();
|
|
|
|
|
|
|
|
return callback();
|
|
|
|
}
|
|
|
|
|
|
|
|
validateCommand(parsed, handler, callback) {
|
|
|
|
let schema = handler.schema || [];
|
|
|
|
let maxArgs = schema.length;
|
|
|
|
let minArgs = schema.filter(item => !item.optional).length;
|
|
|
|
|
|
|
|
// Check if the command can be run in current state
|
|
|
|
if (handler.state && [].concat(handler.state || []).indexOf(this.connection.state) < 0) {
|
|
|
|
return callback(new Error(parsed.command.toUpperCase() + ' not allowed now'));
|
|
|
|
}
|
|
|
|
|
|
|
|
if (handler.schema === false) {
|
|
|
|
//schema check is disabled
|
|
|
|
return callback();
|
|
|
|
}
|
|
|
|
|
|
|
|
// Deny commands with too many arguments
|
|
|
|
if (parsed.attributes && parsed.attributes.length > maxArgs) {
|
|
|
|
return callback(new Error('Too many arguments provided'));
|
|
|
|
}
|
|
|
|
|
|
|
|
// Deny commands with too little arguments
|
|
|
|
if ((parsed.attributes && parsed.attributes.length || 0) < minArgs) {
|
|
|
|
return callback(new Error('Not enough arguments provided'));
|
|
|
|
}
|
|
|
|
|
|
|
|
callback();
|
|
|
|
}
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
module.exports.IMAPCommand = IMAPCommand;
|