wildduck/imap-core/lib/commands/append.js

129 lines
4.2 KiB
JavaScript
Raw Normal View History

2017-03-06 05:45:50 +08:00
'use strict';
2017-06-03 14:51:58 +08:00
const imapTools = require('../imap-tools');
2017-03-06 05:45:50 +08:00
module.exports = {
state: ['Authenticated', 'Selected'],
// we do not show * EXIST response for added message, so keep other notifications quet as well
// otherwise we might end up in situation where APPEND emits an unrelated * EXISTS response
// which does not yet take into account the appended message
disableNotifications: true,
2017-06-03 14:51:58 +08:00
schema: [
{
name: 'path',
2017-06-03 14:51:58 +08:00
type: 'string'
},
{
name: 'flags',
type: 'array',
optional: true
},
{
name: 'datetime',
type: 'string',
optional: true
},
{
name: 'message',
type: 'literal'
}
],
2017-03-06 05:45:50 +08:00
handler(command, callback) {
// Check if APPEND method is set
if (typeof this._server.onAppend !== 'function') {
return callback(null, {
response: 'NO',
message: 'APPEND not implemented'
});
}
let path = Buffer.from((command.attributes.shift() || {}).value || 'binary').toString();
path = imapTools.normalizeMailbox(path, !this.acceptUTF8Enabled);
2017-03-06 05:45:50 +08:00
let message = command.attributes.pop();
let flags = [];
let internaldate = false;
let parsedDate;
if (command.attributes.length === 2) {
flags = command.attributes[0] || [];
2017-06-03 14:51:58 +08:00
internaldate = (command.attributes[1] && command.attributes[1].value) || '';
2017-03-06 05:45:50 +08:00
} else if (command.attributes.length === 1) {
if (Array.isArray(command.attributes[0])) {
flags = command.attributes[0];
} else {
2017-06-03 14:51:58 +08:00
internaldate = (command.attributes[0] && command.attributes[0].value) || '';
2017-03-06 05:45:50 +08:00
}
}
flags = flags.map(flag => (flag.value || '').toString());
if (!path) {
2017-03-06 05:45:50 +08:00
return callback(new Error('Invalid mailbox argument for APPEND'));
}
if (!/^literal$/i.test(message.type)) {
return callback(new Error('Invalid message argument for APPEND'));
}
if (internaldate) {
if (!validateInternalDate(internaldate)) {
return callback(new Error('Invalid date argument for APPEND'));
}
parsedDate = new Date(internaldate);
if (parsedDate.toString() === 'Invalid Date' || parsedDate.getTime() > Date.now() + 24 * 3600 * 1000 || parsedDate.getTime() <= 1000) {
return callback(new Error('Invalid date-time argument for APPEND'));
}
}
for (let i = flags.length - 1; i >= 0; i--) {
if (flags[i].charAt(0) === '\\') {
if (imapTools.systemFlags.indexOf(flags[i].toLowerCase()) < 0) {
return callback(new Error('Invalid system flag argument for APPEND'));
} else {
// fix flag case
flags[i] = flags[i].toLowerCase().replace(/^\\./, c => c.toUpperCase());
}
}
}
// keep only unique flags
flags = flags.filter((flag, i) => {
if (i && flags.slice(0, i).indexOf(flag) >= 0) {
return false;
}
return true;
});
2017-06-03 14:51:58 +08:00
this._server.onAppend(
path,
2017-06-03 14:51:58 +08:00
flags,
internaldate,
new Buffer(typeof message.value === 'string' ? message.value : (message.value || '').toString(), 'binary'),
this.session,
(err, success, info) => {
if (err) {
return callback(err);
}
2017-03-06 05:45:50 +08:00
2017-06-03 14:51:58 +08:00
let code = typeof success === 'string' ? success.toUpperCase() : 'APPENDUID ' + info.uidValidity + ' ' + info.uid;
2017-03-06 05:45:50 +08:00
2017-06-03 14:51:58 +08:00
callback(null, {
response: success === true ? 'OK' : 'NO',
code
});
}
);
2017-03-06 05:45:50 +08:00
}
};
function validateInternalDate(internaldate) {
if (!internaldate || typeof internaldate !== 'string') {
return false;
}
return /^([ \d]\d)-(Jan|Feb|Mar|Apr|May|Jun|Jul|Aug|Sep|Oct|Nov|Dec)-(\d{4}) (\d{2}):(\d{2}):(\d{2}) ([-+])(\d{2})(\d{2})$/i.test(internaldate);
2017-03-06 05:45:50 +08:00
}