diff --git a/lib/Connection.js b/lib/Connection.js index 619862ff..4d161802 100644 --- a/lib/Connection.js +++ b/lib/Connection.js @@ -1,55 +1,64 @@ -var tls = require('tls'), - Socket = require('net').Socket, - EventEmitter = require('events').EventEmitter, - inherits = require('util').inherits, - inspect = require('util').inspect, - isDate = require('util').isDate, - utf7 = require('utf7').imap; - -var Parser = require('./Parser').Parser, - parseExpr = require('./Parser').parseExpr, - parseHeader = require('./Parser').parseHeader; +var tls = require("tls"), + Socket = require("net").Socket, + EventEmitter = require("events").EventEmitter, + inherits = require("util").inherits, + inspect = require("util").inspect, + isDate = require("util").isDate, + utf7 = require("utf7").imap; + +var Parser = require("./Parser").Parser, + parseExpr = require("./Parser").parseExpr, + parseHeader = require("./Parser").parseHeader; var MAX_INT = 9007199254740992, - KEEPALIVE_INTERVAL = 10000, - MAX_IDLE_WAIT = 300000, // 5 minutes - MONTHS = ['Jan', 'Feb', 'Mar', - 'Apr', 'May', 'Jun', - 'Jul', 'Aug', 'Sep', - 'Oct', 'Nov', 'Dec'], - FETCH_ATTR_MAP = { - 'RFC822.SIZE': 'size', - 'BODY': 'struct', - 'BODYSTRUCTURE': 'struct', - 'ENVELOPE': 'envelope', - 'INTERNALDATE': 'date' - }, - SPECIAL_USE_ATTRIBUTES = [ - '\\All', - '\\Archive', - '\\Drafts', - '\\Flagged', - '\\Important', - '\\Junk', - '\\Sent', - '\\Trash' - ], - CRLF = '\r\n', - RE_CMD = /^([^ ]+)(?: |$)/, - RE_UIDCMD_HASRESULTS = /^UID (?:FETCH|SEARCH|SORT)/, - RE_IDLENOOPRES = /^(IDLE|NOOP) /, - RE_OPENBOX = /^EXAMINE|SELECT$/, - RE_BODYPART = /^BODY\[/, - RE_INVALID_KW_CHARS = /[\(\)\{\\\"\]\%\*\x00-\x20\x7F]/, - RE_NUM_RANGE = /^(?:[\d]+|\*):(?:[\d]+|\*)$/, - RE_BACKSLASH = /\\/g, - RE_DBLQUOTE = /"/g, - RE_ESCAPE = /\\\\/g, - RE_INTEGER = /^\d+$/; + KEEPALIVE_INTERVAL = 10000, + MAX_IDLE_WAIT = 300000, // 5 minutes + MONTHS = [ + "Jan", + "Feb", + "Mar", + "Apr", + "May", + "Jun", + "Jul", + "Aug", + "Sep", + "Oct", + "Nov", + "Dec" + ], + FETCH_ATTR_MAP = { + "RFC822.SIZE": "size", + BODY: "struct", + BODYSTRUCTURE: "struct", + ENVELOPE: "envelope", + INTERNALDATE: "date" + }, + SPECIAL_USE_ATTRIBUTES = [ + "\\All", + "\\Archive", + "\\Drafts", + "\\Flagged", + "\\Important", + "\\Junk", + "\\Sent", + "\\Trash" + ], + CRLF = "\r\n", + RE_CMD = /^([^ ]+)(?: |$)/, + RE_UIDCMD_HASRESULTS = /^UID (?:FETCH|SEARCH|SORT)/, + RE_IDLENOOPRES = /^(IDLE|NOOP) /, + RE_OPENBOX = /^EXAMINE|SELECT$/, + RE_BODYPART = /^BODY\[/, + RE_INVALID_KW_CHARS = /[\(\)\{\\\"\]\%\*\x00-\x20\x7F]/, + RE_NUM_RANGE = /^(?:[\d]+|\*):(?:[\d]+|\*)$/, + RE_BACKSLASH = /\\/g, + RE_DBLQUOTE = /"/g, + RE_ESCAPE = /\\\\/g, + RE_INTEGER = /^\d+$/; function Connection(config) { - if (!(this instanceof Connection)) - return new Connection(config); + if (!(this instanceof Connection)) return new Connection(config); EventEmitter.call(this); @@ -59,7 +68,7 @@ function Connection(config) { localAddress: config.localAddress, socket: config.socket, socketTimeout: config.socketTimeout || 0, - host: config.host || 'localhost', + host: config.host || "localhost", port: config.port || 143, tls: config.tls, tlsOptions: config.tlsOptions, @@ -70,9 +79,10 @@ function Connection(config) { xoauth2: config.xoauth2, connTimeout: config.connTimeout || 10000, authTimeout: config.authTimeout || 5000, - keepalive: (config.keepalive === undefined || config.keepalive === null - ? true - : config.keepalive) + keepalive: + config.keepalive === undefined || config.keepalive === null + ? true + : config.keepalive }; this._sock = config.socket || undefined; @@ -87,17 +97,17 @@ function Connection(config) { this._curReq = undefined; this.delimiter = undefined; this.namespaces = undefined; - this.state = 'disconnected'; + this.state = "disconnected"; this.debug = config.debug; } inherits(Connection, EventEmitter); Connection.prototype.connect = function() { var config = this._config, - self = this, - socket, - parser, - tlsOptions; + self = this, + socket, + parser, + tlsOptions; socket = config.socket || new Socket(); socket.setKeepAlive(true); @@ -113,32 +123,34 @@ Connection.prototype.connect = function() { this._curReq = undefined; this.delimiter = undefined; this.namespaces = undefined; - this.state = 'disconnected'; + this.state = "disconnected"; if (config.tls) { tlsOptions = {}; tlsOptions.host = config.host; // Host name may be overridden the tlsOptions - for (var k in config.tlsOptions) - tlsOptions[k] = config.tlsOptions[k]; + for (var k in config.tlsOptions) tlsOptions[k] = config.tlsOptions[k]; tlsOptions.socket = socket; } if (config.tls) - this._sock = tls.connect(tlsOptions, onconnect); + this._sock = tls.connect( + tlsOptions, + onconnect + ); else { - socket.once('connect', onconnect); + socket.once("connect", onconnect); this._sock = socket; } function onconnect() { clearTimeout(self._tmrConn); - self.state = 'connected'; - self.debug && self.debug('[connection] Connected to host'); + self.state = "connected"; + self.debug && self.debug("[connection] Connected to host"); self._tmrAuth = setTimeout(function() { - var err = new Error('Timed out while authenticating with server'); - err.source = 'timeout-auth'; - self.emit('error', err); + var err = new Error("Timed out while authenticating with server"); + err.source = "timeout-auth"; + self.emit("error", err); socket.destroy(); }, config.authTimeout); } @@ -146,55 +158,56 @@ Connection.prototype.connect = function() { this._onError = function(err) { clearTimeout(self._tmrConn); clearTimeout(self._tmrAuth); - self.debug && self.debug('[connection] Error: ' + err); - err.source = 'socket'; - self.emit('error', err); + self.debug && self.debug("[connection] Error: " + err); + err.source = "socket"; + self.emit("error", err); }; - this._sock.on('error', this._onError); + this._sock.on("error", this._onError); this._onSocketTimeout = function() { clearTimeout(self._tmrConn); clearTimeout(self._tmrAuth); clearTimeout(self._tmrKeepalive); - self.state = 'disconnected'; - self.debug && self.debug('[connection] Socket timeout'); + self.state = "disconnected"; + self.debug && self.debug("[connection] Socket timeout"); - var err = new Error('Socket timed out while talking to server'); - err.source = 'socket-timeout'; - self.emit('error', err); + var err = new Error("Socket timed out while talking to server"); + err.source = "socket-timeout"; + self.emit("error", err); socket.destroy(); }; - this._sock.on('timeout', this._onSocketTimeout); + this._sock.on("timeout", this._onSocketTimeout); socket.setTimeout(config.socketTimeout); - socket.once('close', function(had_err) { + socket.once("close", function(had_err) { clearTimeout(self._tmrConn); clearTimeout(self._tmrAuth); clearTimeout(self._tmrKeepalive); - self.state = 'disconnected'; - self.debug && self.debug('[connection] Closed'); - self.emit('close', had_err); + self.state = "disconnected"; + self.debug && self.debug("[connection] Closed"); + self.emit("close", had_err); }); - socket.once('end', function() { + socket.once("end", function() { clearTimeout(self._tmrConn); clearTimeout(self._tmrAuth); clearTimeout(self._tmrKeepalive); - self.state = 'disconnected'; - self.debug && self.debug('[connection] Ended'); - self.emit('end'); + self.state = "disconnected"; + self.debug && self.debug("[connection] Ended"); + self.emit("end"); }); this._parser = parser = new Parser(this._sock, this.debug); - parser.on('untagged', function(info) { + parser.on("untagged", function(info) { self._resUntagged(info); }); - parser.on('tagged', function(info) { + parser.on("tagged", function(info) { self._resTagged(info); }); - parser.on('body', function(stream, info) { - var msg = self._curReq.fetchCache[info.seqno], toget; + parser.on("body", function(stream, info) { + var msg = self._curReq.fetchCache[info.seqno], + toget; if (msg === undefined) { msg = self._curReq.fetchCache[info.seqno] = { @@ -204,7 +217,7 @@ Connection.prototype.connect = function() { ended: false }; - self._curReq.bodyEmitter.emit('message', msg.msgEmitter, info.seqno); + self._curReq.bodyEmitter.emit("message", msg.msgEmitter, info.seqno); } toget = msg.toget; @@ -217,43 +230,47 @@ Connection.prototype.connect = function() { for (var i = 0, len = toget.length; i < len; ++i) { if (_deepEqual(thisbody, toget[i])) { toget.splice(i, 1); - msg.msgEmitter.emit('body', stream, info); + msg.msgEmitter.emit("body", stream, info); return; } } stream.resume(); // a body we didn't ask for? }); - parser.on('continue', function(info) { + parser.on("continue", function(info) { var type = self._curReq.type; - if (type === 'IDLE') { - if (self._queue.length - && self._idle.started === 0 - && self._curReq - && self._curReq.type === 'IDLE' - && self._sock - && self._sock.writable - && !self._idle.enabled) { - self.debug && self.debug('=> DONE'); - self._sock.write('DONE' + CRLF); + if (type === "IDLE") { + if ( + self._queue.length && + self._idle.started === 0 && + self._curReq && + self._curReq.type === "IDLE" && + self._sock && + self._sock.writable && + !self._idle.enabled + ) { + self.debug && self.debug("=> DONE"); + self._sock.write("DONE" + CRLF); return; } // now idling self._idle.started = Date.now(); } else if (/^AUTHENTICATE XOAUTH/.test(self._curReq.fullcmd)) { - self._curReq.oauthError = new Buffer(info.text, 'base64').toString('utf8'); - self.debug && self.debug('=> ' + inspect(CRLF)); + self._curReq.oauthError = new Buffer(info.text, "base64").toString( + "utf8" + ); + self.debug && self.debug("=> " + inspect(CRLF)); self._sock.write(CRLF); - } else if (type === 'APPEND') { + } else if (type === "APPEND") { self._sockWriteAppendData(self._curReq.appendData); } else if (self._curReq.lines && self._curReq.lines.length) { - var line = self._curReq.lines.shift() + '\r\n'; - self.debug && self.debug('=> ' + inspect(line)); - self._sock.write(line, 'binary'); + var line = self._curReq.lines.shift() + "\r\n"; + self.debug && self.debug("=> " + inspect(line)); + self._sock.write(line, "binary"); } }); - parser.on('other', function(line) { + parser.on("other", function(line) { var m; - if (m = RE_IDLENOOPRES.exec(line)) { + if ((m = RE_IDLENOOPRES.exec(line))) { // no longer idling self._idle.enabled = false; self._idle.started = undefined; @@ -261,15 +278,15 @@ Connection.prototype.connect = function() { self._curReq = undefined; - if (self._queue.length === 0 - && self._config.keepalive - && self.state === 'authenticated' - && !self._idle.enabled) { + if ( + self._queue.length === 0 && + self._config.keepalive && + self.state === "authenticated" && + !self._idle.enabled + ) { self._idle.enabled = true; - if (m[1] === 'NOOP') - self._doKeepaliveTimer(); - else - self._doKeepaliveTimer(true); + if (m[1] === "NOOP") self._doKeepaliveTimer(); + else self._doKeepaliveTimer(true); } self._processQueue(); @@ -277,9 +294,9 @@ Connection.prototype.connect = function() { }); this._tmrConn = setTimeout(function() { - var err = new Error('Timed out while connecting to server'); - err.source = 'timeout'; - self.emit('error', err); + var err = new Error("Timed out while connecting to server"); + err.source = "timeout"; + self.emit("error", err); socket.destroy(); }, config.connTimeout); @@ -291,7 +308,7 @@ Connection.prototype.connect = function() { }; Connection.prototype.serverSupports = function(cap) { - return (this._caps && this._caps.indexOf(cap) > -1); + return this._caps && this._caps.indexOf(cap) > -1; }; Connection.prototype.destroy = function() { @@ -302,7 +319,7 @@ Connection.prototype.destroy = function() { Connection.prototype.end = function() { var self = this; - this._enqueue('LOGOUT', function() { + this._enqueue("LOGOUT", function() { self._queue = []; self._curReq = undefined; self._sock.end(); @@ -310,60 +327,55 @@ Connection.prototype.end = function() { }; Connection.prototype.append = function(data, options, cb) { - var literal = this.serverSupports('LITERAL+'); - if (typeof options === 'function') { + var literal = this.serverSupports("LITERAL+"); + if (typeof options === "function") { cb = options; options = undefined; } options = options || {}; if (!options.mailbox) { if (!this._box) - throw new Error('No mailbox specified or currently selected'); - else - options.mailbox = this._box.name; + throw new Error("No mailbox specified or currently selected"); + else options.mailbox = this._box.name; } - var cmd = 'APPEND "' + escape(utf7.encode(''+options.mailbox)) + '"'; + var cmd = 'APPEND "' + escape(utf7.encode("" + options.mailbox)) + '"'; if (options.flags) { - if (!Array.isArray(options.flags)) - options.flags = [options.flags]; + if (!Array.isArray(options.flags)) options.flags = [options.flags]; if (options.flags.length > 0) { for (var i = 0, len = options.flags.length; i < len; ++i) { - if (options.flags[i][0] !== '$' && options.flags[i][0] !== '\\') - options.flags[i] = '\\' + options.flags[i]; + if (options.flags[i][0] !== "$" && options.flags[i][0] !== "\\") + options.flags[i] = "\\" + options.flags[i]; } - cmd += ' (' + options.flags.join(' ') + ')'; + cmd += " (" + options.flags.join(" ") + ")"; } } if (options.date) { - if (!isDate(options.date)) - throw new Error('`date` is not a Date object'); + if (!isDate(options.date)) throw new Error("`date` is not a Date object"); cmd += ' "'; cmd += options.date.getDate(); - cmd += '-'; + cmd += "-"; cmd += MONTHS[options.date.getMonth()]; - cmd += '-'; + cmd += "-"; cmd += options.date.getFullYear(); - cmd += ' '; - cmd += ('0' + options.date.getHours()).slice(-2); - cmd += ':'; - cmd += ('0' + options.date.getMinutes()).slice(-2); - cmd += ':'; - cmd += ('0' + options.date.getSeconds()).slice(-2); - cmd += ((options.date.getTimezoneOffset() > 0) ? ' -' : ' +' ); - cmd += ('0' + (-options.date.getTimezoneOffset() / 60)).slice(-2); - cmd += ('0' + (-options.date.getTimezoneOffset() % 60)).slice(-2); + cmd += " "; + cmd += ("0" + options.date.getHours()).slice(-2); + cmd += ":"; + cmd += ("0" + options.date.getMinutes()).slice(-2); + cmd += ":"; + cmd += ("0" + options.date.getSeconds()).slice(-2); + cmd += options.date.getTimezoneOffset() > 0 ? " -" : " +"; + cmd += ("0" + -options.date.getTimezoneOffset() / 60).slice(-2); + cmd += ("0" + (-options.date.getTimezoneOffset() % 60)).slice(-2); cmd += '"'; } - cmd += ' {'; - cmd += (Buffer.isBuffer(data) ? data.length : Buffer.byteLength(data)); - cmd += (literal ? '+' : '') + '}'; + cmd += " {"; + cmd += Buffer.isBuffer(data) ? data.length : Buffer.byteLength(data); + cmd += (literal ? "+" : "") + "}"; this._enqueue(cmd, cb); - if (literal) - this._queue[this._queue.length - 1].literalAppendData = data; - else - this._queue[this._queue.length - 1].appendData = data; + if (literal) this._queue[this._queue.length - 1].literalAppendData = data; + else this._queue[this._queue.length - 1].appendData = data; }; Connection.prototype.getSpecialUseBoxes = function(cb) { @@ -371,57 +383,54 @@ Connection.prototype.getSpecialUseBoxes = function(cb) { }; Connection.prototype.getBoxes = function(namespace, cb) { - if (typeof namespace === 'function') { + if (typeof namespace === "function") { cb = namespace; - namespace = ''; + namespace = ""; } - namespace = escape(utf7.encode(''+namespace)); + namespace = escape(utf7.encode("" + namespace)); this._enqueue('LIST "' + namespace + '" "*"', cb); }; Connection.prototype.id = function(identification, cb) { - if (!this.serverSupports('ID')) - throw new Error('Server does not support ID'); - var cmd = 'ID'; - if ((identification === null) || (Object.keys(identification).length === 0)) - cmd += ' NIL'; + if (!this.serverSupports("ID")) throw new Error("Server does not support ID"); + var cmd = "ID"; + if (identification === null || Object.keys(identification).length === 0) + cmd += " NIL"; else { if (Object.keys(identification).length > 30) - throw new Error('Max allowed number of keys is 30'); + throw new Error("Max allowed number of keys is 30"); var kv = []; for (var k in identification) { if (Buffer.byteLength(k) > 30) - throw new Error('Max allowed key length is 30'); + throw new Error("Max allowed key length is 30"); if (Buffer.byteLength(identification[k]) > 1024) - throw new Error('Max allowed value length is 1024'); + throw new Error("Max allowed value length is 1024"); kv.push('"' + escape(k) + '"'); kv.push('"' + escape(identification[k]) + '"'); } - cmd += ' (' + kv.join(' ') + ')'; + cmd += " (" + kv.join(" ") + ")"; } this._enqueue(cmd, cb); }; Connection.prototype.openBox = function(name, readOnly, cb) { - if (this.state !== 'authenticated') - throw new Error('Not authenticated'); + if (this.state !== "authenticated") throw new Error("Not authenticated"); - if (typeof readOnly === 'function') { + if (typeof readOnly === "function") { cb = readOnly; readOnly = false; } - name = ''+name; + name = "" + name; var encname = escape(utf7.encode(name)), - cmd = (readOnly ? 'EXAMINE' : 'SELECT'), - self = this; + cmd = readOnly ? "EXAMINE" : "SELECT", + self = this; cmd += ' "' + encname + '"'; - if (this.serverSupports('CONDSTORE')) - cmd += ' (CONDSTORE)'; + if (this.serverSupports("CONDSTORE")) cmd += " (CONDSTORE)"; this._enqueue(cmd, function(err) { if (err) { @@ -436,36 +445,34 @@ Connection.prototype.openBox = function(name, readOnly, cb) { Connection.prototype.closeBox = function(shouldExpunge, cb) { if (this._box === undefined) - throw new Error('No mailbox is currently selected'); + throw new Error("No mailbox is currently selected"); var self = this; - if (typeof shouldExpunge === 'function') { + if (typeof shouldExpunge === "function") { cb = shouldExpunge; shouldExpunge = true; } if (shouldExpunge) { - this._enqueue('CLOSE', function(err) { - if (!err) - self._box = undefined; + this._enqueue("CLOSE", function(err) { + if (!err) self._box = undefined; cb(err); }); } else { - if (this.serverSupports('UNSELECT')) { + if (this.serverSupports("UNSELECT")) { // use UNSELECT if available, as it claims to be "cleaner" than the // alternative "hack" - this._enqueue('UNSELECT', function(err) { - if (!err) - self._box = undefined; + this._enqueue("UNSELECT", function(err) { + if (!err) self._box = undefined; cb(err); }); } else { // "HACK": close the box without expunging by attempting to SELECT a // non-existent mailbox - var badbox = 'NODEJSIMAPCLOSINGBOX' + Date.now(); + var badbox = "NODEJSIMAPCLOSINGBOX" + Date.now(); this._enqueue('SELECT "' + badbox + '"', function(err) { self._box = undefined; cb(); @@ -475,109 +482,105 @@ Connection.prototype.closeBox = function(shouldExpunge, cb) { }; Connection.prototype.addBox = function(name, cb) { - this._enqueue('CREATE "' + escape(utf7.encode(''+name)) + '"', cb); + this._enqueue('CREATE "' + escape(utf7.encode("" + name)) + '"', cb); }; Connection.prototype.delBox = function(name, cb) { - this._enqueue('DELETE "' + escape(utf7.encode(''+name)) + '"', cb); + this._enqueue('DELETE "' + escape(utf7.encode("" + name)) + '"', cb); }; Connection.prototype.renameBox = function(oldname, newname, cb) { - var encoldname = escape(utf7.encode(''+oldname)), - encnewname = escape(utf7.encode(''+newname)), - self = this; - - this._enqueue('RENAME "' + encoldname + '" "' + encnewname + '"', - function(err) { - if (err) - return cb(err); - - if (self._box - && self._box.name === oldname - && oldname.toUpperCase() !== 'INBOX') { - self._box.name = newname; - cb(err, self._box); - } else - cb(); - } - ); + var encoldname = escape(utf7.encode("" + oldname)), + encnewname = escape(utf7.encode("" + newname)), + self = this; + + this._enqueue('RENAME "' + encoldname + '" "' + encnewname + '"', function( + err + ) { + if (err) return cb(err); + + if ( + self._box && + self._box.name === oldname && + oldname.toUpperCase() !== "INBOX" + ) { + self._box.name = newname; + cb(err, self._box); + } else cb(); + }); }; Connection.prototype.subscribeBox = function(name, cb) { - this._enqueue('SUBSCRIBE "' + escape(utf7.encode(''+name)) + '"', cb); + this._enqueue('SUBSCRIBE "' + escape(utf7.encode("" + name)) + '"', cb); }; Connection.prototype.unsubscribeBox = function(name, cb) { - this._enqueue('UNSUBSCRIBE "' + escape(utf7.encode(''+name)) + '"', cb); + this._enqueue('UNSUBSCRIBE "' + escape(utf7.encode("" + name)) + '"', cb); }; Connection.prototype.getSubscribedBoxes = function(namespace, cb) { - if (typeof namespace === 'function') { - cb = namespace; - namespace = ''; - } + if (typeof namespace === "function") { + cb = namespace; + namespace = ""; + } - namespace = escape(utf7.encode(''+namespace)); + namespace = escape(utf7.encode("" + namespace)); - this._enqueue('LSUB "' + namespace + '" "*"', cb); + this._enqueue('LSUB "' + namespace + '" "*"', cb); }; Connection.prototype.status = function(boxName, cb) { if (this._box && this._box.name === boxName) - throw new Error('Cannot call status on currently selected mailbox'); + throw new Error("Cannot call status on currently selected mailbox"); - boxName = escape(utf7.encode(''+boxName)); + boxName = escape(utf7.encode("" + boxName)); - var info = [ 'MESSAGES', 'RECENT', 'UNSEEN', 'UIDVALIDITY', 'UIDNEXT' ]; + var info = ["MESSAGES", "RECENT", "UNSEEN", "UIDVALIDITY", "UIDNEXT"]; - if (this.serverSupports('CONDSTORE')) - info.push('HIGHESTMODSEQ'); + if (this.serverSupports("CONDSTORE")) info.push("HIGHESTMODSEQ"); - info = info.join(' '); + info = info.join(" "); - this._enqueue('STATUS "' + boxName + '" (' + info + ')', cb); + this._enqueue('STATUS "' + boxName + '" (' + info + ")", cb); }; Connection.prototype.expunge = function(uids, cb) { - if (typeof uids === 'function') { + if (typeof uids === "function") { cb = uids; uids = undefined; } if (uids !== undefined) { - if (!Array.isArray(uids)) - uids = [uids]; + if (!Array.isArray(uids)) uids = [uids]; validateUIDList(uids); - if (uids.length === 0) - throw new Error('Empty uid list'); + if (uids.length === 0) throw new Error("Empty uid list"); - uids = uids.join(','); + uids = uids.join(","); - if (!this.serverSupports('UIDPLUS')) - throw new Error('Server does not support this feature (UIDPLUS)'); + if (!this.serverSupports("UIDPLUS")) + throw new Error("Server does not support this feature (UIDPLUS)"); - this._enqueue('UID EXPUNGE ' + uids, cb); - } else - this._enqueue('EXPUNGE', cb); + this._enqueue("UID EXPUNGE " + uids, cb); + } else this._enqueue("EXPUNGE", cb); }; Connection.prototype.search = function(criteria, cb) { - this._search('UID ', criteria, cb); + this._search("UID ", criteria, cb); }; Connection.prototype._search = function(which, criteria, cb) { if (this._box === undefined) - throw new Error('No mailbox is currently selected'); + throw new Error("No mailbox is currently selected"); else if (!Array.isArray(criteria)) - throw new Error('Expected array for search criteria'); + throw new Error("Expected array for search criteria"); - var cmd = which + 'SEARCH', - info = { hasUTF8: false /*output*/ }, - query = buildSearchQuery(criteria, this._caps, info), - lines; + var cmd = which + "SEARCH", + info = { hasUTF8: false /*output*/ }, + query = buildSearchQuery(criteria, this._caps, info), + lines; if (info.hasUTF8) { - cmd += ' CHARSET UTF-8'; + cmd += " CHARSET UTF-8"; lines = query.split(CRLF); query = lines.shift(); } @@ -590,137 +593,149 @@ Connection.prototype._search = function(which, criteria, cb) { }; Connection.prototype.addFlags = function(uids, flags, cb) { - this._store('UID ', uids, { mode: '+', flags: flags }, cb); + this._store("UID ", uids, { mode: "+", flags: flags }, cb); }; Connection.prototype.delFlags = function(uids, flags, cb) { - this._store('UID ', uids, { mode: '-', flags: flags }, cb); + this._store("UID ", uids, { mode: "-", flags: flags }, cb); }; Connection.prototype.setFlags = function(uids, flags, cb) { - this._store('UID ', uids, { mode: '', flags: flags }, cb); + this._store("UID ", uids, { mode: "", flags: flags }, cb); }; Connection.prototype.addKeywords = function(uids, keywords, cb) { - this._store('UID ', uids, { mode: '+', keywords: keywords }, cb); + this._store("UID ", uids, { mode: "+", keywords: keywords }, cb); }; Connection.prototype.delKeywords = function(uids, keywords, cb) { - this._store('UID ', uids, { mode: '-', keywords: keywords }, cb); + this._store("UID ", uids, { mode: "-", keywords: keywords }, cb); }; Connection.prototype.setKeywords = function(uids, keywords, cb) { - this._store('UID ', uids, { mode: '', keywords: keywords }, cb); + this._store("UID ", uids, { mode: "", keywords: keywords }, cb); }; Connection.prototype._store = function(which, uids, cfg, cb) { var mode = cfg.mode, - isFlags = (cfg.flags !== undefined), - items = (isFlags ? cfg.flags : cfg.keywords); + isFlags = cfg.flags !== undefined, + items = isFlags ? cfg.flags : cfg.keywords; if (this._box === undefined) - throw new Error('No mailbox is currently selected'); - else if (uids === undefined) - throw new Error('No messages specified'); + throw new Error("No mailbox is currently selected"); + else if (uids === undefined) throw new Error("No messages specified"); - if (!Array.isArray(uids)) - uids = [uids]; + if (!Array.isArray(uids)) uids = [uids]; validateUIDList(uids); if (uids.length === 0) { - throw new Error('Empty ' - + (which === '' ? 'sequence number' : 'uid') - + 'list'); + throw new Error( + "Empty " + (which === "" ? "sequence number" : "uid") + "list" + ); } - if ((!Array.isArray(items) && typeof items !== 'string') - || (Array.isArray(items) && items.length === 0)) - throw new Error((isFlags ? 'Flags' : 'Keywords') - + ' argument must be a string or a non-empty Array'); - if (!Array.isArray(items)) - items = [items]; + if ( + (!Array.isArray(items) && typeof items !== "string") || + (Array.isArray(items) && items.length === 0) + ) + throw new Error( + (isFlags ? "Flags" : "Keywords") + + " argument must be a string or a non-empty Array" + ); + if (!Array.isArray(items)) items = [items]; for (var i = 0, len = items.length; i < len; ++i) { if (isFlags) { - if (items[i][0] !== '\\') - items[i] = '\\' + items[i]; + if (items[i][0] !== "\\") items[i] = "\\" + items[i]; } else { // keyword contains any char except control characters (%x00-1F and %x7F) // and: '(', ')', '{', ' ', '%', '*', '\', '"', ']' if (RE_INVALID_KW_CHARS.test(items[i])) { - throw new Error('The keyword "' + items[i] - + '" contains invalid characters'); + throw new Error( + 'The keyword "' + items[i] + '" contains invalid characters' + ); } } } - items = items.join(' '); - uids = uids.join(','); + items = items.join(" "); + uids = uids.join(","); - var modifiers = ''; + var modifiers = ""; if (cfg.modseq !== undefined && !this._box.nomodseq) - modifiers += 'UNCHANGEDSINCE ' + cfg.modseq + ' '; - - this._enqueue(which + 'STORE ' + uids + ' ' - + modifiers - + mode + 'FLAGS.SILENT (' + items + ')', cb); + modifiers += "UNCHANGEDSINCE " + cfg.modseq + " "; + + this._enqueue( + which + + "STORE " + + uids + + " " + + modifiers + + mode + + "FLAGS.SILENT (" + + items + + ")", + cb + ); }; Connection.prototype.copy = function(uids, boxTo, cb) { - this._copy('UID ', uids, boxTo, cb); + this._copy("UID ", uids, boxTo, cb); }; Connection.prototype._copy = function(which, uids, boxTo, cb) { if (this._box === undefined) - throw new Error('No mailbox is currently selected'); + throw new Error("No mailbox is currently selected"); - if (!Array.isArray(uids)) - uids = [uids]; + if (!Array.isArray(uids)) uids = [uids]; validateUIDList(uids); if (uids.length === 0) { - throw new Error('Empty ' - + (which === '' ? 'sequence number' : 'uid') - + 'list'); + throw new Error( + "Empty " + (which === "" ? "sequence number" : "uid") + "list" + ); } - boxTo = escape(utf7.encode(''+boxTo)); + boxTo = escape(utf7.encode("" + boxTo)); - this._enqueue(which + 'COPY ' + uids.join(',') + ' "' + boxTo + '"', cb); + this._enqueue(which + "COPY " + uids.join(",") + ' "' + boxTo + '"', cb); }; Connection.prototype.move = function(uids, boxTo, cb) { - this._move('UID ', uids, boxTo, cb); + this._move("UID ", uids, boxTo, cb); }; Connection.prototype._move = function(which, uids, boxTo, cb) { if (this._box === undefined) - throw new Error('No mailbox is currently selected'); + throw new Error("No mailbox is currently selected"); - if (this.serverSupports('MOVE')) { - if (!Array.isArray(uids)) - uids = [uids]; + if (this.serverSupports("MOVE")) { + if (!Array.isArray(uids)) uids = [uids]; validateUIDList(uids); if (uids.length === 0) { - throw new Error('Empty ' - + (which === '' ? 'sequence number' : 'uid') - + 'list'); + throw new Error( + "Empty " + (which === "" ? "sequence number" : "uid") + "list" + ); } - uids = uids.join(','); - boxTo = escape(utf7.encode(''+boxTo)); - - this._enqueue(which + 'MOVE ' + uids + ' "' + boxTo + '"', cb); - } else if (this._box.permFlags.indexOf('\\Deleted') === -1 - && this._box.flags.indexOf('\\Deleted') === -1) { - throw new Error('Cannot move message: ' - + 'server does not allow deletion of messages'); + uids = uids.join(","); + boxTo = escape(utf7.encode("" + boxTo)); + + this._enqueue(which + "MOVE " + uids + ' "' + boxTo + '"', cb); + } else if ( + this._box.permFlags.indexOf("\\Deleted") === -1 && + this._box.flags.indexOf("\\Deleted") === -1 + ) { + throw new Error( + "Cannot move message: " + "server does not allow deletion of messages" + ); } else { - var deletedUIDs, task = 0, self = this; + var deletedUIDs, + task = 0, + self = this; this._copy(which, uids, boxTo, function ccb(err, info) { - if (err) - return cb(err, info); + if (err) return cb(err, info); - if (task === 0 && which && self.serverSupports('UIDPLUS')) { + if (task === 0 && which && self.serverSupports("UIDPLUS")) { // UIDPLUS gives us a 'UID EXPUNGE n' command to expunge a subset of // messages with the \Deleted flag set. This allows us to skip some // actions. @@ -729,14 +744,14 @@ Connection.prototype._move = function(which, uids, boxTo, cb) { // Make sure we don't expunge any messages marked as Deleted except the // one we are moving if (task === 0) { - self.search(['DELETED'], function(e, result) { + self.search(["DELETED"], function(e, result) { ++task; deletedUIDs = result; ccb(e, info); }); } else if (task === 1) { if (deletedUIDs.length) { - self.delFlags(deletedUIDs, '\\Deleted', function(e) { + self.delFlags(deletedUIDs, "\\Deleted", function(e) { ++task; ccb(e, info); }); @@ -749,12 +764,10 @@ Connection.prototype._move = function(which, uids, boxTo, cb) { ++task; ccb(e, info); }; - if (which) - self.addFlags(uids, '\\Deleted', cbMarkDel); - else - self.seq.addFlags(uids, '\\Deleted', cbMarkDel); + if (which) self.addFlags(uids, "\\Deleted", cbMarkDel); + else self.seq.addFlags(uids, "\\Deleted", cbMarkDel); } else if (task === 3) { - if (which && self.serverSupports('UIDPLUS')) { + if (which && self.serverSupports("UIDPLUS")) { self.expunge(uids, function(e) { cb(e, info); }); @@ -766,96 +779,96 @@ Connection.prototype._move = function(which, uids, boxTo, cb) { } } else if (task === 4) { if (deletedUIDs.length) { - self.addFlags(deletedUIDs, '\\Deleted', function(e) { + self.addFlags(deletedUIDs, "\\Deleted", function(e) { cb(e, info); }); - } else - cb(err, info); + } else cb(err, info); } }); } }; Connection.prototype.fetch = function(uids, options) { - return this._fetch('UID ', uids, options); + return this._fetch("UID ", uids, options); }; Connection.prototype._fetch = function(which, uids, options) { - if (uids === undefined - || uids === null - || (Array.isArray(uids) && uids.length === 0)) - throw new Error('Nothing to fetch'); - - if (!Array.isArray(uids)) - uids = [uids]; + if ( + uids === undefined || + uids === null || + (Array.isArray(uids) && uids.length === 0) + ) + throw new Error("Nothing to fetch"); + + if (!Array.isArray(uids)) uids = [uids]; validateUIDList(uids); if (uids.length === 0) { - throw new Error('Empty ' - + (which === '' ? 'sequence number' : 'uid') - + 'list'); + throw new Error( + "Empty " + (which === "" ? "sequence number" : "uid") + "list" + ); } - uids = uids.join(','); + uids = uids.join(","); - var cmd = which + 'FETCH ' + uids + ' (', - fetching = [], - i, len, key; + var cmd = which + "FETCH " + uids + " (", + fetching = [], + i, + len, + key; - if (this.serverSupports('X-GM-EXT-1')) { - fetching.push('X-GM-THRID'); - fetching.push('X-GM-MSGID'); - fetching.push('X-GM-LABELS'); + if (this.serverSupports("X-GM-EXT-1")) { + fetching.push("X-GM-THRID"); + fetching.push("X-GM-MSGID"); + fetching.push("X-GM-LABELS"); } - if (this.serverSupports('CONDSTORE') && !this._box.nomodseq) - fetching.push('MODSEQ'); + if (this.serverSupports("CONDSTORE") && !this._box.nomodseq) + fetching.push("MODSEQ"); - fetching.push('UID'); - fetching.push('FLAGS'); - fetching.push('INTERNALDATE'); + fetching.push("UID"); + fetching.push("FLAGS"); + fetching.push("INTERNALDATE"); var modifiers; if (options) { modifiers = options.modifiers; - if (options.envelope) - fetching.push('ENVELOPE'); - if (options.struct) - fetching.push('BODYSTRUCTURE'); - if (options.size) - fetching.push('RFC822.SIZE'); + if (options.envelope) fetching.push("ENVELOPE"); + if (options.struct) fetching.push("BODYSTRUCTURE"); + if (options.size) fetching.push("RFC822.SIZE"); if (Array.isArray(options.extensions)) { - options.extensions.forEach(function (extension) { + options.extensions.forEach(function(extension) { fetching.push(extension.toUpperCase()); }); } - cmd += fetching.join(' '); + cmd += fetching.join(" "); if (options.bodies !== undefined) { var bodies = options.bodies, - prefix = (options.markSeen ? '' : '.PEEK'); - if (!Array.isArray(bodies)) - bodies = [bodies]; + prefix = options.markSeen ? "" : ".PEEK"; + if (!Array.isArray(bodies)) bodies = [bodies]; for (i = 0, len = bodies.length; i < len; ++i) { - fetching.push(parseExpr(''+bodies[i])); - cmd += ' BODY' + prefix + '[' + bodies[i] + ']'; + fetching.push(parseExpr("" + bodies[i])); + cmd += " BODY" + prefix + "[" + bodies[i] + "]"; } } - } else - cmd += fetching.join(' '); + } else cmd += fetching.join(" "); - cmd += ')'; + cmd += ")"; - var modkeys = (typeof modifiers === 'object' ? Object.keys(modifiers) : []), - modstr = ' ('; + var modkeys = typeof modifiers === "object" ? Object.keys(modifiers) : [], + modstr = " ("; for (i = 0, len = modkeys.length, key; i < len; ++i) { key = modkeys[i].toUpperCase(); - if (key === 'CHANGEDSINCE' && this.serverSupports('CONDSTORE') - && !this._box.nomodseq) - modstr += key + ' ' + modifiers[modkeys[i]] + ' '; + if ( + key === "CHANGEDSINCE" && + this.serverSupports("CONDSTORE") && + !this._box.nomodseq + ) + modstr += key + " " + modifiers[modkeys[i]] + " "; } if (modstr.length > 2) { cmd += modstr.substring(0, modstr.length - 1); - cmd += ')'; + cmd += ")"; } this._enqueue(cmd); @@ -867,104 +880,129 @@ Connection.prototype._fetch = function(which, uids, options) { // Extension methods =========================================================== Connection.prototype.setLabels = function(uids, labels, cb) { - this._storeLabels('UID ', uids, labels, '', cb); + this._storeLabels("UID ", uids, labels, "", cb); }; Connection.prototype.addLabels = function(uids, labels, cb) { - this._storeLabels('UID ', uids, labels, '+', cb); + this._storeLabels("UID ", uids, labels, "+", cb); }; Connection.prototype.delLabels = function(uids, labels, cb) { - this._storeLabels('UID ', uids, labels, '-', cb); + this._storeLabels("UID ", uids, labels, "-", cb); }; Connection.prototype._storeLabels = function(which, uids, labels, mode, cb) { - if (!this.serverSupports('X-GM-EXT-1')) - throw new Error('Server must support X-GM-EXT-1 capability'); + if (!this.serverSupports("X-GM-EXT-1")) + throw new Error("Server must support X-GM-EXT-1 capability"); else if (this._box === undefined) - throw new Error('No mailbox is currently selected'); - else if (uids === undefined) - throw new Error('No messages specified'); + throw new Error("No mailbox is currently selected"); + else if (uids === undefined) throw new Error("No messages specified"); - if (!Array.isArray(uids)) - uids = [uids]; + if (!Array.isArray(uids)) uids = [uids]; validateUIDList(uids); if (uids.length === 0) { - throw new Error('Empty ' - + (which === '' ? 'sequence number' : 'uid') - + 'list'); + throw new Error( + "Empty " + (which === "" ? "sequence number" : "uid") + "list" + ); } - if ((!Array.isArray(labels) && typeof labels !== 'string') - || (Array.isArray(labels) && labels.length === 0)) - throw new Error('labels argument must be a string or a non-empty Array'); - - if (!Array.isArray(labels)) - labels = [labels]; - labels = labels.map(function(v) { - return '"' + escape(utf7.encode(''+v)) + '"'; - }).join(' '); - - uids = uids.join(','); - - this._enqueue(which + 'STORE ' + uids + ' ' + mode - + 'X-GM-LABELS.SILENT (' + labels + ')', cb); + if ( + (!Array.isArray(labels) && typeof labels !== "string") || + (Array.isArray(labels) && labels.length === 0) + ) + throw new Error("labels argument must be a string or a non-empty Array"); + + if (!Array.isArray(labels)) labels = [labels]; + labels = labels + .map(function(v) { + return '"' + escape(utf7.encode("" + v)) + '"'; + }) + .join(" "); + + uids = uids.join(","); + + this._enqueue( + which + + "STORE " + + uids + + " " + + mode + + "X-GM-LABELS.SILENT (" + + labels + + ")", + cb + ); }; Connection.prototype.sort = function(sorts, criteria, cb) { - this._sort('UID ', sorts, criteria, cb); + this._sort("UID ", sorts, criteria, cb); }; Connection.prototype._sort = function(which, sorts, criteria, cb) { + let displaySupported = false; if (this._box === undefined) - throw new Error('No mailbox is currently selected'); + throw new Error("No mailbox is currently selected"); else if (!Array.isArray(sorts) || !sorts.length) - throw new Error('Expected array with at least one sort criteria'); + throw new Error("Expected array with at least one sort criteria"); else if (!Array.isArray(criteria)) - throw new Error('Expected array for search criteria'); - else if (!this.serverSupports('SORT')) - throw new Error('Sort is not supported on the server'); - - sorts = sorts.map(function(c) { - if (typeof c !== 'string') - throw new Error('Unexpected sort criteria data type. ' - + 'Expected string. Got: ' + typeof criteria); - - var modifier = ''; - if (c[0] === '-') { - modifier = 'REVERSE '; + throw new Error("Expected array for search criteria"); + else if (!this.serverSupports("SORT")) + throw new Error("Sort is not supported on the server"); + + if(this.serverSupports("SORT=DISPLAY")){ + displaySupported = true; + } + + + sorts = sorts.map(function(c, this) { + if (typeof c !== "string") + throw new Error( + "Unexpected sort criteria data type. " + + "Expected string. Got: " + + typeof criteria + ); + + var modifier = ""; + if (c[0] === "-") { + modifier = "REVERSE "; c = c.substring(1); } switch (c.toUpperCase()) { - case 'ARRIVAL': - case 'CC': - case 'DATE': - case 'FROM': - case 'SIZE': - case 'SUBJECT': - case 'TO': + case "ARRIVAL": + case "CC": + case "DATE": + case "FROM": + case "SIZE": + case "SUBJECT": + case "TO": + break; + case "DISPLAYFROM": + case "DISPLAYTO": + if (!displaySupported) { + throw new Error("Unexpected sort criteria: " + c); + } break; default: - throw new Error('Unexpected sort criteria: ' + c); + throw new Error("Unexpected sort criteria: " + c); } return modifier + c; }); - sorts = sorts.join(' '); + sorts = sorts.join(" "); var info = { hasUTF8: false /*output*/ }, - query = buildSearchQuery(criteria, this._caps, info), - charset = 'US-ASCII', - lines; + query = buildSearchQuery(criteria, this._caps, info), + charset = "US-ASCII", + lines; if (info.hasUTF8) { - charset = 'UTF-8'; + charset = "UTF-8"; lines = query.split(CRLF); query = lines.shift(); } - this._enqueue(which + 'SORT (' + sorts + ') ' + charset + query, cb); + this._enqueue(which + "SORT (" + sorts + ") " + charset + query, cb); if (info.hasUTF8) { var req = this._queue[this._queue.length - 1]; req.lines = lines; @@ -972,35 +1010,36 @@ Connection.prototype._sort = function(which, sorts, criteria, cb) { }; Connection.prototype.esearch = function(criteria, options, cb) { - this._esearch('UID ', criteria, options, cb); + this._esearch("UID ", criteria, options, cb); }; Connection.prototype._esearch = function(which, criteria, options, cb) { if (this._box === undefined) - throw new Error('No mailbox is currently selected'); + throw new Error("No mailbox is currently selected"); else if (!Array.isArray(criteria)) - throw new Error('Expected array for search options'); + throw new Error("Expected array for search options"); var info = { hasUTF8: false /*output*/ }, - query = buildSearchQuery(criteria, this._caps, info), - charset = '', - lines; + query = buildSearchQuery(criteria, this._caps, info), + charset = "", + lines; if (info.hasUTF8) { - charset = ' CHARSET UTF-8'; + charset = " CHARSET UTF-8"; lines = query.split(CRLF); query = lines.shift(); } - if (typeof options === 'function') { + if (typeof options === "function") { cb = options; - options = ''; - } else if (!options) - options = ''; + options = ""; + } else if (!options) options = ""; - if (Array.isArray(options)) - options = options.join(' '); + if (Array.isArray(options)) options = options.join(" "); - this._enqueue(which + 'SEARCH RETURN (' + options + ')' + charset + query, cb); + this._enqueue( + which + "SEARCH RETURN (" + options + ")" + charset + query, + cb + ); if (info.hasUTF8) { var req = this._queue[this._queue.length - 1]; req.lines = lines; @@ -1008,47 +1047,44 @@ Connection.prototype._esearch = function(which, criteria, options, cb) { }; Connection.prototype.setQuota = function(quotaRoot, limits, cb) { - if (typeof limits === 'function') { + if (typeof limits === "function") { cb = limits; limits = {}; } - var triplets = ''; + var triplets = ""; for (var l in limits) { - if (triplets) - triplets += ' '; - triplets += l + ' ' + limits[l]; + if (triplets) triplets += " "; + triplets += l + " " + limits[l]; } - quotaRoot = escape(utf7.encode(''+quotaRoot)); + quotaRoot = escape(utf7.encode("" + quotaRoot)); - this._enqueue('SETQUOTA "' + quotaRoot + '" (' + triplets + ')', - function(err, quotalist) { - if (err) - return cb(err); + this._enqueue('SETQUOTA "' + quotaRoot + '" (' + triplets + ")", function( + err, + quotalist + ) { + if (err) return cb(err); - cb(err, quotalist ? quotalist[0] : limits); - } - ); + cb(err, quotalist ? quotalist[0] : limits); + }); }; Connection.prototype.getQuota = function(quotaRoot, cb) { - quotaRoot = escape(utf7.encode(''+quotaRoot)); + quotaRoot = escape(utf7.encode("" + quotaRoot)); this._enqueue('GETQUOTA "' + quotaRoot + '"', function(err, quotalist) { - if (err) - return cb(err); + if (err) return cb(err); cb(err, quotalist[0]); }); }; Connection.prototype.getQuotaRoot = function(boxName, cb) { - boxName = escape(utf7.encode(''+boxName)); + boxName = escape(utf7.encode("" + boxName)); this._enqueue('GETQUOTAROOT "' + boxName + '"', function(err, quotalist) { - if (err) - return cb(err); + if (err) return cb(err); var quotas = {}; if (quotalist) { @@ -1061,26 +1097,26 @@ Connection.prototype.getQuotaRoot = function(boxName, cb) { }; Connection.prototype.thread = function(algorithm, criteria, cb) { - this._thread('UID ', algorithm, criteria, cb); + this._thread("UID ", algorithm, criteria, cb); }; Connection.prototype._thread = function(which, algorithm, criteria, cb) { algorithm = algorithm.toUpperCase(); - if (!this.serverSupports('THREAD=' + algorithm)) - throw new Error('Server does not support that threading algorithm'); + if (!this.serverSupports("THREAD=" + algorithm)) + throw new Error("Server does not support that threading algorithm"); var info = { hasUTF8: false /*output*/ }, - query = buildSearchQuery(criteria, this._caps, info), - charset = 'US-ASCII', - lines; + query = buildSearchQuery(criteria, this._caps, info), + charset = "US-ASCII", + lines; if (info.hasUTF8) { - charset = 'UTF-8'; + charset = "UTF-8"; lines = query.split(CRLF); query = lines.shift(); } - this._enqueue(which + 'THREAD ' + algorithm + ' ' + charset + query, cb); + this._enqueue(which + "THREAD " + algorithm + " " + charset + query, cb); if (info.hasUTF8) { var req = this._queue[this._queue.length - 1]; req.lines = lines; @@ -1088,261 +1124,261 @@ Connection.prototype._thread = function(which, algorithm, criteria, cb) { }; Connection.prototype.addFlagsSince = function(uids, flags, modseq, cb) { - this._store('UID ', - uids, - { mode: '+', flags: flags, modseq: modseq }, - cb); + this._store("UID ", uids, { mode: "+", flags: flags, modseq: modseq }, cb); }; Connection.prototype.delFlagsSince = function(uids, flags, modseq, cb) { - this._store('UID ', - uids, - { mode: '-', flags: flags, modseq: modseq }, - cb); + this._store("UID ", uids, { mode: "-", flags: flags, modseq: modseq }, cb); }; Connection.prototype.setFlagsSince = function(uids, flags, modseq, cb) { - this._store('UID ', - uids, - { mode: '', flags: flags, modseq: modseq }, - cb); + this._store("UID ", uids, { mode: "", flags: flags, modseq: modseq }, cb); }; Connection.prototype.addKeywordsSince = function(uids, keywords, modseq, cb) { - this._store('UID ', - uids, - { mode: '+', keywords: keywords, modseq: modseq }, - cb); + this._store( + "UID ", + uids, + { mode: "+", keywords: keywords, modseq: modseq }, + cb + ); }; Connection.prototype.delKeywordsSince = function(uids, keywords, modseq, cb) { - this._store('UID ', - uids, - { mode: '-', keywords: keywords, modseq: modseq }, - cb); + this._store( + "UID ", + uids, + { mode: "-", keywords: keywords, modseq: modseq }, + cb + ); }; Connection.prototype.setKeywordsSince = function(uids, keywords, modseq, cb) { - this._store('UID ', - uids, - { mode: '', keywords: keywords, modseq: modseq }, - cb); + this._store( + "UID ", + uids, + { mode: "", keywords: keywords, modseq: modseq }, + cb + ); }; // END Extension methods ======================================================= // Namespace for seqno-based commands -Object.defineProperty(Connection.prototype, 'seq', { get: function() { - var self = this; - return { - delKeywords: function(seqnos, keywords, cb) { - self._store('', seqnos, { mode: '-', keywords: keywords }, cb); - }, - addKeywords: function(seqnos, keywords, cb) { - self._store('', seqnos, { mode: '+', keywords: keywords }, cb); - }, - setKeywords: function(seqnos, keywords, cb) { - self._store('', seqnos, { mode: '', keywords: keywords }, cb); - }, +Object.defineProperty(Connection.prototype, "seq", { + get: function() { + var self = this; + return { + delKeywords: function(seqnos, keywords, cb) { + self._store("", seqnos, { mode: "-", keywords: keywords }, cb); + }, + addKeywords: function(seqnos, keywords, cb) { + self._store("", seqnos, { mode: "+", keywords: keywords }, cb); + }, + setKeywords: function(seqnos, keywords, cb) { + self._store("", seqnos, { mode: "", keywords: keywords }, cb); + }, - delFlags: function(seqnos, flags, cb) { - self._store('', seqnos, { mode: '-', flags: flags }, cb); - }, - addFlags: function(seqnos, flags, cb) { - self._store('', seqnos, { mode: '+', flags: flags }, cb); - }, - setFlags: function(seqnos, flags, cb) { - self._store('', seqnos, { mode: '', flags: flags }, cb); - }, + delFlags: function(seqnos, flags, cb) { + self._store("", seqnos, { mode: "-", flags: flags }, cb); + }, + addFlags: function(seqnos, flags, cb) { + self._store("", seqnos, { mode: "+", flags: flags }, cb); + }, + setFlags: function(seqnos, flags, cb) { + self._store("", seqnos, { mode: "", flags: flags }, cb); + }, - move: function(seqnos, boxTo, cb) { - self._move('', seqnos, boxTo, cb); - }, - copy: function(seqnos, boxTo, cb) { - self._copy('', seqnos, boxTo, cb); - }, - fetch: function(seqnos, options) { - return self._fetch('', seqnos, options); - }, - search: function(options, cb) { - self._search('', options, cb); - }, + move: function(seqnos, boxTo, cb) { + self._move("", seqnos, boxTo, cb); + }, + copy: function(seqnos, boxTo, cb) { + self._copy("", seqnos, boxTo, cb); + }, + fetch: function(seqnos, options) { + return self._fetch("", seqnos, options); + }, + search: function(options, cb) { + self._search("", options, cb); + }, - // Extensions ============================================================== - delLabels: function(seqnos, labels, cb) { - self._storeLabels('', seqnos, labels, '-', cb); - }, - addLabels: function(seqnos, labels, cb) { - self._storeLabels('', seqnos, labels, '+', cb); - }, - setLabels: function(seqnos, labels, cb) { - self._storeLabels('', seqnos, labels, '', cb); - }, + // Extensions ============================================================== + delLabels: function(seqnos, labels, cb) { + self._storeLabels("", seqnos, labels, "-", cb); + }, + addLabels: function(seqnos, labels, cb) { + self._storeLabels("", seqnos, labels, "+", cb); + }, + setLabels: function(seqnos, labels, cb) { + self._storeLabels("", seqnos, labels, "", cb); + }, - esearch: function(criteria, options, cb) { - self._esearch('', criteria, options, cb); - }, + esearch: function(criteria, options, cb) { + self._esearch("", criteria, options, cb); + }, - sort: function(sorts, options, cb) { - self._sort('', sorts, options, cb); - }, - thread: function(algorithm, criteria, cb) { - self._thread('', algorithm, criteria, cb); - }, + sort: function(sorts, options, cb) { + self._sort("", sorts, options, cb); + }, + thread: function(algorithm, criteria, cb) { + self._thread("", algorithm, criteria, cb); + }, - delKeywordsSince: function(seqnos, keywords, modseq, cb) { - self._store('', - seqnos, - { mode: '-', keywords: keywords, modseq: modseq }, - cb); - }, - addKeywordsSince: function(seqnos, keywords, modseq, cb) { - self._store('', - seqnos, - { mode: '+', keywords: keywords, modseq: modseq }, - cb); - }, - setKeywordsSince: function(seqnos, keywords, modseq, cb) { - self._store('', - seqnos, - { mode: '', keywords: keywords, modseq: modseq }, - cb); - }, + delKeywordsSince: function(seqnos, keywords, modseq, cb) { + self._store( + "", + seqnos, + { mode: "-", keywords: keywords, modseq: modseq }, + cb + ); + }, + addKeywordsSince: function(seqnos, keywords, modseq, cb) { + self._store( + "", + seqnos, + { mode: "+", keywords: keywords, modseq: modseq }, + cb + ); + }, + setKeywordsSince: function(seqnos, keywords, modseq, cb) { + self._store( + "", + seqnos, + { mode: "", keywords: keywords, modseq: modseq }, + cb + ); + }, - delFlagsSince: function(seqnos, flags, modseq, cb) { - self._store('', - seqnos, - { mode: '-', flags: flags, modseq: modseq }, - cb); - }, - addFlagsSince: function(seqnos, flags, modseq, cb) { - self._store('', - seqnos, - { mode: '+', flags: flags, modseq: modseq }, - cb); - }, - setFlagsSince: function(seqnos, flags, modseq, cb) { - self._store('', - seqnos, - { mode: '', flags: flags, modseq: modseq }, - cb); - } - }; -}}); + delFlagsSince: function(seqnos, flags, modseq, cb) { + self._store( + "", + seqnos, + { mode: "-", flags: flags, modseq: modseq }, + cb + ); + }, + addFlagsSince: function(seqnos, flags, modseq, cb) { + self._store( + "", + seqnos, + { mode: "+", flags: flags, modseq: modseq }, + cb + ); + }, + setFlagsSince: function(seqnos, flags, modseq, cb) { + self._store("", seqnos, { mode: "", flags: flags, modseq: modseq }, cb); + } + }; + } +}); Connection.prototype._resUntagged = function(info) { - var type = info.type, i, len, box, attrs, key; - - if (type === 'bye') - this._sock.end(); - else if (type === 'namespace') - this.namespaces = info.text; - else if (type === 'id') - this._curReq.cbargs.push(info.text); - else if (type === 'capability') - this._caps = info.text.map(function(v) { return v.toUpperCase(); }); - else if (type === 'preauth') - this.state = 'authenticated'; - else if (type === 'sort' || type === 'thread' || type === 'esearch') + var type = info.type, + i, + len, + box, + attrs, + key; + + if (type === "bye") this._sock.end(); + else if (type === "namespace") this.namespaces = info.text; + else if (type === "id") this._curReq.cbargs.push(info.text); + else if (type === "capability") + this._caps = info.text.map(function(v) { + return v.toUpperCase(); + }); + else if (type === "preauth") this.state = "authenticated"; + else if (type === "sort" || type === "thread" || type === "esearch") this._curReq.cbargs.push(info.text); - else if (type === 'search') { + else if (type === "search") { if (info.text.results !== undefined) { // CONDSTORE-modified search results this._curReq.cbargs.push(info.text.results); this._curReq.cbargs.push(info.text.modseq); - } else - this._curReq.cbargs.push(info.text); - } else if (type === 'quota') { + } else this._curReq.cbargs.push(info.text); + } else if (type === "quota") { var cbargs = this._curReq.cbargs; - if (!cbargs.length) - cbargs.push([]); + if (!cbargs.length) cbargs.push([]); cbargs[0].push(info.text); - } else if (type === 'recent') { + } else if (type === "recent") { if (!this._box && RE_OPENBOX.test(this._curReq.type)) this._createCurrentBox(); - if (this._box) - this._box.messages.new = info.num; - } else if (type === 'flags') { + if (this._box) this._box.messages.new = info.num; + } else if (type === "flags") { if (!this._box && RE_OPENBOX.test(this._curReq.type)) this._createCurrentBox(); - if (this._box) - this._box.flags = info.text; - } else if (type === 'bad' || type === 'no') { - if (this.state === 'connected' && !this._curReq) { + if (this._box) this._box.flags = info.text; + } else if (type === "bad" || type === "no") { + if (this.state === "connected" && !this._curReq) { clearTimeout(this._tmrConn); clearTimeout(this._tmrAuth); - var err = new Error('Received negative welcome: ' + info.text); - err.source = 'protocol'; - this.emit('error', err); + var err = new Error("Received negative welcome: " + info.text); + err.source = "protocol"; + this.emit("error", err); this._sock.end(); } - } else if (type === 'exists') { + } else if (type === "exists") { if (!this._box && RE_OPENBOX.test(this._curReq.type)) this._createCurrentBox(); if (this._box) { var prev = this._box.messages.total, - now = info.num; + now = info.num; this._box.messages.total = now; - if (now > prev && this.state === 'authenticated') { + if (now > prev && this.state === "authenticated") { this._box.messages.new = now - prev; - this.emit('mail', this._box.messages.new); + this.emit("mail", this._box.messages.new); } } - } else if (type === 'expunge') { + } else if (type === "expunge") { if (this._box) { - if (this._box.messages.total > 0) - --this._box.messages.total; - this.emit('expunge', info.num); + if (this._box.messages.total > 0) --this._box.messages.total; + this.emit("expunge", info.num); } - } else if (type === 'ok') { - if (this.state === 'connected' && !this._curReq) - this._login(); - else if (typeof info.textCode === 'string' - && info.textCode.toUpperCase() === 'ALERT') - this.emit('alert', info.text); - else if (this._curReq - && info.textCode - && (RE_OPENBOX.test(this._curReq.type))) { + } else if (type === "ok") { + if (this.state === "connected" && !this._curReq) this._login(); + else if ( + typeof info.textCode === "string" && + info.textCode.toUpperCase() === "ALERT" + ) + this.emit("alert", info.text); + else if ( + this._curReq && + info.textCode && + RE_OPENBOX.test(this._curReq.type) + ) { // we're opening a mailbox - if (!this._box) - this._createCurrentBox(); - - if (info.textCode.key) - key = info.textCode.key.toUpperCase(); - else - key = info.textCode; - - if (key === 'UIDVALIDITY') - this._box.uidvalidity = info.textCode.val; - else if (key === 'UIDNEXT') - this._box.uidnext = info.textCode.val; - else if (key === 'HIGHESTMODSEQ') - this._box.highestmodseq = ''+info.textCode.val; - else if (key === 'PERMANENTFLAGS') { + if (!this._box) this._createCurrentBox(); + + if (info.textCode.key) key = info.textCode.key.toUpperCase(); + else key = info.textCode; + + if (key === "UIDVALIDITY") this._box.uidvalidity = info.textCode.val; + else if (key === "UIDNEXT") this._box.uidnext = info.textCode.val; + else if (key === "HIGHESTMODSEQ") + this._box.highestmodseq = "" + info.textCode.val; + else if (key === "PERMANENTFLAGS") { var idx, permFlags, keywords; this._box.permFlags = permFlags = info.textCode.val; - if ((idx = this._box.permFlags.indexOf('\\*')) > -1) { + if ((idx = this._box.permFlags.indexOf("\\*")) > -1) { this._box.newKeywords = true; permFlags.splice(idx, 1); } this._box.keywords = keywords = permFlags.filter(function(f) { - return (f[0] !== '\\'); - }); + return f[0] !== "\\"; + }); for (i = 0, len = keywords.length; i < len; ++i) permFlags.splice(permFlags.indexOf(keywords[i]), 1); - } else if (key === 'UIDNOTSTICKY') - this._box.persistentUIDs = false; - else if (key === 'NOMODSEQ') - this._box.nomodseq = true; - } else if (typeof info.textCode === 'string' - && info.textCode.toUpperCase() === 'UIDVALIDITY') - this.emit('uidvalidity', info.text); - } else if (type === 'list' || type === 'lsub' || type === 'xlist') { - if (this.delimiter === undefined) - this.delimiter = info.text.delimiter; + } else if (key === "UIDNOTSTICKY") this._box.persistentUIDs = false; + else if (key === "NOMODSEQ") this._box.nomodseq = true; + } else if ( + typeof info.textCode === "string" && + info.textCode.toUpperCase() === "UIDVALIDITY" + ) + this.emit("uidvalidity", info.text); + } else if (type === "list" || type === "lsub" || type === "xlist") { + if (this.delimiter === undefined) this.delimiter = info.text.delimiter; else { - if (this._curReq.cbargs.length === 0) - this._curReq.cbargs.push({}); + if (this._curReq.cbargs.length === 0) this._curReq.cbargs.push({}); box = { attribs: info.text.flags, @@ -1356,15 +1392,14 @@ Connection.prototype._resUntagged = function(info) { box.special_use_attrib = SPECIAL_USE_ATTRIBUTES[i]; var name = info.text.name, - curChildren = this._curReq.cbargs[0]; + curChildren = this._curReq.cbargs[0]; if (box.delimiter) { var path = name.split(box.delimiter), - parent = null; + parent = null; name = path.pop(); for (i = 0, len = path.length; i < len; ++i) { - if (!curChildren[path[i]]) - curChildren[path[i]] = {}; + if (!curChildren[path[i]]) curChildren[path[i]] = {}; if (!curChildren[path[i]].children) curChildren[path[i]].children = {}; parent = curChildren[path[i]]; @@ -1372,11 +1407,10 @@ Connection.prototype._resUntagged = function(info) { } box.parent = parent; } - if (curChildren[name]) - box.children = curChildren[name].children; + if (curChildren[name]) box.children = curChildren[name].children; curChildren[name] = box; } - } else if (type === 'status') { + } else if (type === "status") { box = { name: info.text.name, uidnext: 0, @@ -1390,38 +1424,35 @@ Connection.prototype._resUntagged = function(info) { attrs = info.text.attrs; if (attrs) { - if (attrs.recent !== undefined) - box.messages.new = attrs.recent; - if (attrs.unseen !== undefined) - box.messages.unseen = attrs.unseen; - if (attrs.messages !== undefined) - box.messages.total = attrs.messages; - if (attrs.uidnext !== undefined) - box.uidnext = attrs.uidnext; - if (attrs.uidvalidity !== undefined) - box.uidvalidity = attrs.uidvalidity; - if (attrs.highestmodseq !== undefined) // CONDSTORE - box.highestmodseq = ''+attrs.highestmodseq; + if (attrs.recent !== undefined) box.messages.new = attrs.recent; + if (attrs.unseen !== undefined) box.messages.unseen = attrs.unseen; + if (attrs.messages !== undefined) box.messages.total = attrs.messages; + if (attrs.uidnext !== undefined) box.uidnext = attrs.uidnext; + if (attrs.uidvalidity !== undefined) box.uidvalidity = attrs.uidvalidity; + if (attrs.highestmodseq !== undefined) + // CONDSTORE + box.highestmodseq = "" + attrs.highestmodseq; } this._curReq.cbargs.push(box); - } else if (type === 'fetch') { + } else if (type === "fetch") { if (/^(?:UID )?FETCH/.test(this._curReq.fullcmd)) { // FETCH response sent as result of FETCH request var msg = this._curReq.fetchCache[info.num], - keys = Object.keys(info.text), - keyslen = keys.length, - toget, msgEmitter, j; + keys = Object.keys(info.text), + keyslen = keys.length, + toget, + msgEmitter, + j; if (msg === undefined) { // simple case -- no bodies were streamed toget = this._curReq.fetching.slice(0); - if (toget.length === 0) - return; + if (toget.length === 0) return; msgEmitter = new EventEmitter(); attrs = {}; - this._curReq.bodyEmitter.emit('message', msgEmitter, info.num); + this._curReq.bodyEmitter.emit("message", msgEmitter, info.num); } else { toget = msg.toget; msgEmitter = msg.msgEmitter; @@ -1433,7 +1464,7 @@ Connection.prototype._resUntagged = function(info) { if (msg && !msg.ended) { msg.ended = true; process.nextTick(function() { - msgEmitter.emit('end'); + msgEmitter.emit("end"); }); } return; @@ -1445,14 +1476,13 @@ Connection.prototype._resUntagged = function(info) { while (--j >= 0) { if (keys[j].toUpperCase() === toget[i]) { if (!RE_BODYPART.test(toget[i])) { - if (toget[i] === 'X-GM-LABELS') { + if (toget[i] === "X-GM-LABELS") { var labels = info.text[keys[j]]; for (var k = 0, lenk = labels.length; k < lenk; ++k) - labels[k] = (''+labels[k]).replace(RE_ESCAPE, '\\'); + labels[k] = ("" + labels[k]).replace(RE_ESCAPE, "\\"); } key = FETCH_ATTR_MAP[toget[i]]; - if (!key) - key = toget[i].toLowerCase(); + if (!key) key = toget[i].toLowerCase(); attrs[key] = info.text[keys[j]]; } toget.splice(i, 1); @@ -1463,11 +1493,10 @@ Connection.prototype._resUntagged = function(info) { } if (toget.length === 0) { - if (msg) - msg.ended = true; + if (msg) msg.ended = true; process.nextTick(function() { - msgEmitter.emit('attributes', attrs); - msgEmitter.emit('end'); + msgEmitter.emit("attributes", attrs); + msgEmitter.emit("end"); }); } else if (msg === undefined) { this._curReq.fetchCache[info.num] = { @@ -1480,33 +1509,32 @@ Connection.prototype._resUntagged = function(info) { } else { // FETCH response sent as result of STORE request or sent unilaterally, // treat them as the same for now for simplicity - this.emit('update', info.num, info.text); + this.emit("update", info.num, info.text); } } }; Connection.prototype._resTagged = function(info) { - var req = this._curReq, err; + var req = this._curReq, + err; - if (!req) - return; + if (!req) return; this._curReq = undefined; - if (info.type === 'no' || info.type === 'bad') { + if (info.type === "no" || info.type === "bad") { var errtext; - if (info.text) - errtext = info.text; - else - errtext = req.oauthError; + if (info.text) errtext = info.text; + else errtext = req.oauthError; err = new Error(errtext); err.type = info.type; err.textCode = info.textCode; - err.source = 'protocol'; + err.source = "protocol"; } else if (this._box) { - if (req.type === 'EXAMINE' || req.type === 'SELECT') { - this._box.readOnly = (typeof info.textCode === 'string' - && info.textCode.toUpperCase() === 'READ-ONLY'); + if (req.type === "EXAMINE" || req.type === "SELECT") { + this._box.readOnly = + typeof info.textCode === "string" && + info.textCode.toUpperCase() === "READ-ONLY"; } // According to RFC 3501, UID commands do not give errors for @@ -1518,27 +1546,30 @@ Connection.prototype._resTagged = function(info) { if (req.bodyEmitter) { var bodyEmitter = req.bodyEmitter; - if (err) - bodyEmitter.emit('error', err); + if (err) bodyEmitter.emit("error", err); process.nextTick(function() { - bodyEmitter.emit('end'); + bodyEmitter.emit("end"); }); } else { req.cbargs.unshift(err); if (info.textCode && info.textCode.key) { var key = info.textCode.key.toUpperCase(); - if (key === 'APPENDUID') // [uidvalidity, newUID] + if (key === "APPENDUID") + // [uidvalidity, newUID] req.cbargs.push(info.textCode.val[1]); - else if (key === 'COPYUID') // [uidvalidity, sourceUIDs, destUIDs] + else if (key === "COPYUID") + // [uidvalidity, sourceUIDs, destUIDs] req.cbargs.push(info.textCode.val[2]); } req.cb && req.cb.apply(this, req.cbargs); } - if (this._queue.length === 0 - && this._config.keepalive - && this.state === 'authenticated' - && !this._idle.enabled) { + if ( + this._queue.length === 0 && + this._config.keepalive && + this.state === "authenticated" && + !this._idle.enabled + ) { this._idle.enabled = true; this._doKeepaliveTimer(true); } @@ -1548,7 +1579,7 @@ Connection.prototype._resTagged = function(info) { Connection.prototype._createCurrentBox = function() { this._box = { - name: '', + name: "", flags: [], readOnly: false, uidvalidity: 0, @@ -1567,135 +1598,144 @@ Connection.prototype._createCurrentBox = function() { Connection.prototype._doKeepaliveTimer = function(immediate) { var self = this, - interval = this._config.keepalive.interval || KEEPALIVE_INTERVAL, - idleWait = this._config.keepalive.idleInterval || MAX_IDLE_WAIT, - forceNoop = this._config.keepalive.forceNoop || false, - timerfn = function() { - if (self._idle.enabled) { - // unlike NOOP, IDLE is only a valid command after authenticating - if (!self.serverSupports('IDLE') - || self.state !== 'authenticated' - || forceNoop) - self._enqueue('NOOP', true); - else { - if (self._idle.started === undefined) { - self._idle.started = 0; - self._enqueue('IDLE', true); - } else if (self._idle.started > 0) { - var timeDiff = Date.now() - self._idle.started; - if (timeDiff >= idleWait) { - self._idle.enabled = false; - self.debug && self.debug('=> DONE'); - self._sock.write('DONE' + CRLF); - return; - } + interval = this._config.keepalive.interval || KEEPALIVE_INTERVAL, + idleWait = this._config.keepalive.idleInterval || MAX_IDLE_WAIT, + forceNoop = this._config.keepalive.forceNoop || false, + timerfn = function() { + if (self._idle.enabled) { + // unlike NOOP, IDLE is only a valid command after authenticating + if ( + !self.serverSupports("IDLE") || + self.state !== "authenticated" || + forceNoop + ) + self._enqueue("NOOP", true); + else { + if (self._idle.started === undefined) { + self._idle.started = 0; + self._enqueue("IDLE", true); + } else if (self._idle.started > 0) { + var timeDiff = Date.now() - self._idle.started; + if (timeDiff >= idleWait) { + self._idle.enabled = false; + self.debug && self.debug("=> DONE"); + self._sock.write("DONE" + CRLF); + return; } - self._tmrKeepalive = setTimeout(timerfn, interval); } + self._tmrKeepalive = setTimeout(timerfn, interval); } - }; + } + }; - if (immediate) - timerfn(); - else - this._tmrKeepalive = setTimeout(timerfn, interval); + if (immediate) timerfn(); + else this._tmrKeepalive = setTimeout(timerfn, interval); }; Connection.prototype._login = function() { - var self = this, checkedNS = false; + var self = this, + checkedNS = false; var reentry = function(err) { clearTimeout(self._tmrAuth); if (err) { - self.emit('error', err); + self.emit("error", err); return self._sock.end(); } // 2. Get the list of available namespaces (RFC2342) - if (!checkedNS && self.serverSupports('NAMESPACE')) { + if (!checkedNS && self.serverSupports("NAMESPACE")) { checkedNS = true; - return self._enqueue('NAMESPACE', reentry); + return self._enqueue("NAMESPACE", reentry); } // 3. Get the top-level mailbox hierarchy delimiter used by the server self._enqueue('LIST "" ""', function() { - self.state = 'authenticated'; - self.emit('ready'); + self.state = "authenticated"; + self.emit("ready"); }); }; // 1. Get the supported capabilities - self._enqueue('CAPABILITY', function() { + self._enqueue("CAPABILITY", function() { // No need to attempt the login sequence if we're on a PREAUTH connection. - if (self.state === 'connected') { + if (self.state === "connected") { var err, - checkCaps = function(error) { - if (error) { - error.source = 'authentication'; - return reentry(error); - } + checkCaps = function(error) { + if (error) { + error.source = "authentication"; + return reentry(error); + } - if (self._caps === undefined) { - // Fetch server capabilities if they were not automatically - // provided after authentication - return self._enqueue('CAPABILITY', reentry); - } else - reentry(); - }; - - if (self.serverSupports('STARTTLS') - && (self._config.autotls === 'always' - || (self._config.autotls === 'required' - && self.serverSupports('LOGINDISABLED')))) { - self._starttls(); - return; + if (self._caps === undefined) { + // Fetch server capabilities if they were not automatically + // provided after authentication + return self._enqueue("CAPABILITY", reentry); + } else reentry(); + }; + + if ( + self.serverSupports("STARTTLS") && + (self._config.autotls === "always" || + (self._config.autotls === "required" && + self.serverSupports("LOGINDISABLED"))) + ) { + self._starttls(); + return; } - if (self.serverSupports('LOGINDISABLED')) { - err = new Error('Logging in is disabled on this server'); - err.source = 'authentication'; + if (self.serverSupports("LOGINDISABLED")) { + err = new Error("Logging in is disabled on this server"); + err.source = "authentication"; return reentry(err); } var cmd; - if (self.serverSupports('AUTH=XOAUTH') && self._config.xoauth) { + if (self.serverSupports("AUTH=XOAUTH") && self._config.xoauth) { self._caps = undefined; - cmd = 'AUTHENTICATE XOAUTH'; + cmd = "AUTHENTICATE XOAUTH"; // are there any servers that support XOAUTH/XOAUTH2 and not SASL-IR? //if (self.serverSupports('SASL-IR')) - cmd += ' ' + escape(self._config.xoauth); + cmd += " " + escape(self._config.xoauth); self._enqueue(cmd, checkCaps); - } else if (self.serverSupports('AUTH=XOAUTH2') && self._config.xoauth2) { + } else if (self.serverSupports("AUTH=XOAUTH2") && self._config.xoauth2) { self._caps = undefined; - cmd = 'AUTHENTICATE XOAUTH2'; + cmd = "AUTHENTICATE XOAUTH2"; //if (self.serverSupports('SASL-IR')) - cmd += ' ' + escape(self._config.xoauth2); + cmd += " " + escape(self._config.xoauth2); self._enqueue(cmd, checkCaps); } else if (self._config.user && self._config.password) { self._caps = undefined; - self._enqueue('LOGIN "' + escape(self._config.user) + '" "' - + escape(self._config.password) + '"', checkCaps); + self._enqueue( + 'LOGIN "' + + escape(self._config.user) + + '" "' + + escape(self._config.password) + + '"', + checkCaps + ); } else { - err = new Error('No supported authentication method(s) available. ' - + 'Unable to login.'); - err.source = 'authentication'; + err = new Error( + "No supported authentication method(s) available. " + + "Unable to login." + ); + err.source = "authentication"; return reentry(err); } - } else - reentry(); + } else reentry(); }); }; Connection.prototype._starttls = function() { var self = this; - this._enqueue('STARTTLS', function(err) { + this._enqueue("STARTTLS", function(err) { if (err) { - self.emit('error', err); + self.emit("error", err); return self._sock.end(); } self._caps = undefined; - self._sock.removeAllListeners('error'); + self._sock.removeAllListeners("error"); var tlsOptions = {}; @@ -1705,12 +1745,15 @@ Connection.prototype._starttls = function() { tlsOptions[k] = this._config.tlsOptions[k]; tlsOptions.socket = self._sock; - self._sock = tls.connect(tlsOptions, function() { - self._login(); - }); + self._sock = tls.connect( + tlsOptions, + function() { + self._login(); + } + ); - self._sock.on('error', self._onError); - self._sock.on('timeout', self._onSocketTimeout); + self._sock.on("error", self._onError); + self._sock.on("timeout", self._onSocketTimeout); self._sock.setTimeout(self._config.socketTimeout); self._parser.setStream(self._sock); @@ -1718,24 +1761,27 @@ Connection.prototype._starttls = function() { }; Connection.prototype._processQueue = function() { - if (this._curReq || !this._queue.length || !this._sock || !this._sock.writable) + if ( + this._curReq || + !this._queue.length || + !this._sock || + !this._sock.writable + ) return; this._curReq = this._queue.shift(); - if (this._tagcount === MAX_INT) - this._tagcount = 0; + if (this._tagcount === MAX_INT) this._tagcount = 0; var prefix; - if (this._curReq.type === 'IDLE' || this._curReq.type === 'NOOP') + if (this._curReq.type === "IDLE" || this._curReq.type === "NOOP") prefix = this._curReq.type; - else - prefix = 'A' + (this._tagcount++); + else prefix = "A" + this._tagcount++; - var out = prefix + ' ' + this._curReq.fullcmd; - this.debug && this.debug('=> ' + inspect(out)); - this._sock.write(out + CRLF, 'utf8'); + var out = prefix + " " + this._curReq.fullcmd; + this.debug && this.debug("=> " + inspect(out)); + this._sock.write(out + CRLF, "utf8"); if (this._curReq.literalAppendData) { // LITERAL+: we are appending a mesage, and not waiting for a reply @@ -1743,53 +1789,55 @@ Connection.prototype._processQueue = function() { } }; -Connection.prototype._sockWriteAppendData = function(appendData) -{ +Connection.prototype._sockWriteAppendData = function(appendData) { var val = appendData; - if (Buffer.isBuffer(appendData)) - val = val.toString('utf8'); + if (Buffer.isBuffer(appendData)) val = val.toString("utf8"); - this.debug && this.debug('=> ' + inspect(val)); + this.debug && this.debug("=> " + inspect(val)); this._sock.write(val); this._sock.write(CRLF); }; Connection.prototype._enqueue = function(fullcmd, promote, cb) { - if (typeof promote === 'function') { + if (typeof promote === "function") { cb = promote; promote = false; } var info = { - type: fullcmd.match(RE_CMD)[1], - fullcmd: fullcmd, - cb: cb, - cbargs: [] - }, - self = this; + type: fullcmd.match(RE_CMD)[1], + fullcmd: fullcmd, + cb: cb, + cbargs: [] + }, + self = this; - if (promote) - this._queue.unshift(info); - else - this._queue.push(info); + if (promote) this._queue.unshift(info); + else this._queue.push(info); - if (!this._curReq - && this.state !== 'disconnected' - && this.state !== 'upgrading') { + if ( + !this._curReq && + this.state !== "disconnected" && + this.state !== "upgrading" + ) { // defer until next tick for requests like APPEND and FETCH where access to // the request object is needed immediately after enqueueing - process.nextTick(function() { self._processQueue(); }); - } else if (this._curReq - && this._curReq.type === 'IDLE' - && this._sock - && this._sock.writable - && this._idle.enabled) { + process.nextTick(function() { + self._processQueue(); + }); + } else if ( + this._curReq && + this._curReq.type === "IDLE" && + this._sock && + this._sock.writable && + this._idle.enabled + ) { this._idle.enabled = false; clearTimeout(this._tmrKeepalive); if (this._idle.started > 0) { // we've seen the continuation for our IDLE - this.debug && this.debug('=> DONE'); - this._sock.write('DONE' + CRLF); + this.debug && this.debug("=> DONE"); + this._sock.write("DONE" + CRLF); } } }; @@ -1801,34 +1849,29 @@ module.exports = Connection; // utilities ------------------------------------------------------------------- function escape(str) { - return str.replace(RE_BACKSLASH, '\\\\').replace(RE_DBLQUOTE, '\\"'); + return str.replace(RE_BACKSLASH, "\\\\").replace(RE_DBLQUOTE, '\\"'); } function validateUIDList(uids, noThrow) { for (var i = 0, len = uids.length, intval; i < len; ++i) { - if (typeof uids[i] === 'string') { - if (uids[i] === '*' || uids[i] === '*:*') { - if (len > 1) - uids = ['*']; + if (typeof uids[i] === "string") { + if (uids[i] === "*" || uids[i] === "*:*") { + if (len > 1) uids = ["*"]; break; - } else if (RE_NUM_RANGE.test(uids[i])) - continue; + } else if (RE_NUM_RANGE.test(uids[i])) continue; } - intval = parseInt(''+uids[i], 10); + intval = parseInt("" + uids[i], 10); if (isNaN(intval)) { - var err = new Error('UID/seqno must be an integer, "*", or a range: ' - + uids[i]); - if (noThrow) - return err; - else - throw err; + var err = new Error( + 'UID/seqno must be an integer, "*", or a range: ' + uids[i] + ); + if (noThrow) return err; + else throw err; } else if (intval <= 0) { - var err = new Error('UID/seqno must be greater than zero'); - if (noThrow) - return err; - else - throw err; - } else if (typeof uids[i] !== 'number') { + var err = new Error("UID/seqno must be greater than zero"); + if (noThrow) return err; + else throw err; + } else if (typeof uids[i] !== "number") { uids[i] = intval; } } @@ -1836,198 +1879,206 @@ function validateUIDList(uids, noThrow) { function hasNonASCII(str) { for (var i = 0, len = str.length; i < len; ++i) { - if (str.charCodeAt(i) > 0x7F) - return true; + if (str.charCodeAt(i) > 0x7f) return true; } return false; } function buildString(str) { - if (typeof str !== 'string') - str = ''+str; + if (typeof str !== "string") str = "" + str; if (hasNonASCII(str)) { - var buf = new Buffer(str, 'utf8'); - return '{' + buf.length + '}\r\n' + buf.toString('binary'); - } else - return '"' + escape(str) + '"'; + var buf = new Buffer(str, "utf8"); + return "{" + buf.length + "}\r\n" + buf.toString("binary"); + } else return '"' + escape(str) + '"'; } function buildSearchQuery(options, extensions, info, isOrChild) { - var searchargs = '', err, val; + var searchargs = "", + err, + val; for (var i = 0, len = options.length; i < len; ++i) { - var criteria = (isOrChild ? options : options[i]), - args = null, - modifier = (isOrChild ? '' : ' '); - if (typeof criteria === 'string') - criteria = criteria.toUpperCase(); + var criteria = isOrChild ? options : options[i], + args = null, + modifier = isOrChild ? "" : " "; + if (typeof criteria === "string") criteria = criteria.toUpperCase(); else if (Array.isArray(criteria)) { - if (criteria.length > 1) - args = criteria.slice(1); - if (criteria.length > 0) - criteria = criteria[0].toUpperCase(); + if (criteria.length > 1) args = criteria.slice(1); + if (criteria.length > 0) criteria = criteria[0].toUpperCase(); } else - throw new Error('Unexpected search option data type. ' - + 'Expected string or array. Got: ' + typeof criteria); - if (criteria === 'OR') { + throw new Error( + "Unexpected search option data type. " + + "Expected string or array. Got: " + + typeof criteria + ); + if (criteria === "OR") { if (args.length !== 2) - throw new Error('OR must have exactly two arguments'); - if (isOrChild) - searchargs += 'OR ('; - else - searchargs += ' OR ('; + throw new Error("OR must have exactly two arguments"); + if (isOrChild) searchargs += "OR ("; + else searchargs += " OR ("; searchargs += buildSearchQuery(args[0], extensions, info, true); - searchargs += ') ('; + searchargs += ") ("; searchargs += buildSearchQuery(args[1], extensions, info, true); - searchargs += ')'; + searchargs += ")"; } else { - if (criteria[0] === '!') { - modifier += 'NOT '; + if (criteria[0] === "!") { + modifier += "NOT "; criteria = criteria.substr(1); } - switch(criteria) { + switch (criteria) { // -- Standard criteria -- - case 'ALL': - case 'ANSWERED': - case 'DELETED': - case 'DRAFT': - case 'FLAGGED': - case 'NEW': - case 'SEEN': - case 'RECENT': - case 'OLD': - case 'UNANSWERED': - case 'UNDELETED': - case 'UNDRAFT': - case 'UNFLAGGED': - case 'UNSEEN': + case "ALL": + case "ANSWERED": + case "DELETED": + case "DRAFT": + case "FLAGGED": + case "NEW": + case "SEEN": + case "RECENT": + case "OLD": + case "UNANSWERED": + case "UNDELETED": + case "UNDRAFT": + case "UNFLAGGED": + case "UNSEEN": searchargs += modifier + criteria; - break; - case 'BCC': - case 'BODY': - case 'CC': - case 'FROM': - case 'SUBJECT': - case 'TEXT': - case 'TO': + break; + case "BCC": + case "BODY": + case "CC": + case "FROM": + case "SUBJECT": + case "TEXT": + case "TO": if (!args || args.length !== 1) - throw new Error('Incorrect number of arguments for search option: ' - + criteria); + throw new Error( + "Incorrect number of arguments for search option: " + criteria + ); val = buildString(args[0]); - if (info && val[0] === '{') - info.hasUTF8 = true; - searchargs += modifier + criteria + ' ' + val; - break; - case 'BEFORE': - case 'ON': - case 'SENTBEFORE': - case 'SENTON': - case 'SENTSINCE': - case 'SINCE': + if (info && val[0] === "{") info.hasUTF8 = true; + searchargs += modifier + criteria + " " + val; + break; + case "BEFORE": + case "ON": + case "SENTBEFORE": + case "SENTON": + case "SENTSINCE": + case "SINCE": if (!args || args.length !== 1) - throw new Error('Incorrect number of arguments for search option: ' - + criteria); + throw new Error( + "Incorrect number of arguments for search option: " + criteria + ); else if (!(args[0] instanceof Date)) { - if ((args[0] = new Date(args[0])).toString() === 'Invalid Date') - throw new Error('Search option argument must be a Date object' - + ' or a parseable date string'); + if ((args[0] = new Date(args[0])).toString() === "Invalid Date") + throw new Error( + "Search option argument must be a Date object" + + " or a parseable date string" + ); } - searchargs += modifier + criteria + ' ' + args[0].getDate() + '-' - + MONTHS[args[0].getMonth()] + '-' - + args[0].getFullYear(); - break; - case 'KEYWORD': - case 'UNKEYWORD': + searchargs += + modifier + + criteria + + " " + + args[0].getDate() + + "-" + + MONTHS[args[0].getMonth()] + + "-" + + args[0].getFullYear(); + break; + case "KEYWORD": + case "UNKEYWORD": if (!args || args.length !== 1) - throw new Error('Incorrect number of arguments for search option: ' - + criteria); - searchargs += modifier + criteria + ' ' + args[0]; - break; - case 'LARGER': - case 'SMALLER': + throw new Error( + "Incorrect number of arguments for search option: " + criteria + ); + searchargs += modifier + criteria + " " + args[0]; + break; + case "LARGER": + case "SMALLER": if (!args || args.length !== 1) - throw new Error('Incorrect number of arguments for search option: ' - + criteria); + throw new Error( + "Incorrect number of arguments for search option: " + criteria + ); var num = parseInt(args[0], 10); if (isNaN(num)) - throw new Error('Search option argument must be a number'); - searchargs += modifier + criteria + ' ' + args[0]; - break; - case 'HEADER': + throw new Error("Search option argument must be a number"); + searchargs += modifier + criteria + " " + args[0]; + break; + case "HEADER": if (!args || args.length !== 2) - throw new Error('Incorrect number of arguments for search option: ' - + criteria); + throw new Error( + "Incorrect number of arguments for search option: " + criteria + ); val = buildString(args[1]); - if (info && val[0] === '{') - info.hasUTF8 = true; - searchargs += modifier + criteria + ' "' + escape(''+args[0]) - + '" ' + val; - break; - case 'UID': + if (info && val[0] === "{") info.hasUTF8 = true; + searchargs += + modifier + criteria + ' "' + escape("" + args[0]) + '" ' + val; + break; + case "UID": if (!args) - throw new Error('Incorrect number of arguments for search option: ' - + criteria); + throw new Error( + "Incorrect number of arguments for search option: " + criteria + ); validateUIDList(args); - if (args.length === 0) - throw new Error('Empty uid list'); - searchargs += modifier + criteria + ' ' + args.join(','); - break; + if (args.length === 0) throw new Error("Empty uid list"); + searchargs += modifier + criteria + " " + args.join(","); + break; // Extensions ========================================================== - case 'X-GM-MSGID': // Gmail unique message ID - case 'X-GM-THRID': // Gmail thread ID - if (extensions.indexOf('X-GM-EXT-1') === -1) - throw new Error('IMAP extension not available for: ' + criteria); + case "X-GM-MSGID": // Gmail unique message ID + case "X-GM-THRID": // Gmail thread ID + if (extensions.indexOf("X-GM-EXT-1") === -1) + throw new Error("IMAP extension not available for: " + criteria); if (!args || args.length !== 1) - throw new Error('Incorrect number of arguments for search option: ' - + criteria); + throw new Error( + "Incorrect number of arguments for search option: " + criteria + ); else { - val = ''+args[0]; - if (!(RE_INTEGER.test(args[0]))) - throw new Error('Invalid value'); + val = "" + args[0]; + if (!RE_INTEGER.test(args[0])) throw new Error("Invalid value"); } - searchargs += modifier + criteria + ' ' + val; - break; - case 'X-GM-RAW': // Gmail search syntax - if (extensions.indexOf('X-GM-EXT-1') === -1) - throw new Error('IMAP extension not available for: ' + criteria); + searchargs += modifier + criteria + " " + val; + break; + case "X-GM-RAW": // Gmail search syntax + if (extensions.indexOf("X-GM-EXT-1") === -1) + throw new Error("IMAP extension not available for: " + criteria); if (!args || args.length !== 1) - throw new Error('Incorrect number of arguments for search option: ' - + criteria); + throw new Error( + "Incorrect number of arguments for search option: " + criteria + ); val = buildString(args[0]); - if (info && val[0] === '{') - info.hasUTF8 = true; - searchargs += modifier + criteria + ' ' + val; - break; - case 'X-GM-LABELS': // Gmail labels - if (extensions.indexOf('X-GM-EXT-1') === -1) - throw new Error('IMAP extension not available for: ' + criteria); + if (info && val[0] === "{") info.hasUTF8 = true; + searchargs += modifier + criteria + " " + val; + break; + case "X-GM-LABELS": // Gmail labels + if (extensions.indexOf("X-GM-EXT-1") === -1) + throw new Error("IMAP extension not available for: " + criteria); if (!args || args.length !== 1) - throw new Error('Incorrect number of arguments for search option: ' - + criteria); - searchargs += modifier + criteria + ' ' + args[0]; - break; - case 'MODSEQ': - if (extensions.indexOf('CONDSTORE') === -1) - throw new Error('IMAP extension not available for: ' + criteria); + throw new Error( + "Incorrect number of arguments for search option: " + criteria + ); + searchargs += modifier + criteria + " " + args[0]; + break; + case "MODSEQ": + if (extensions.indexOf("CONDSTORE") === -1) + throw new Error("IMAP extension not available for: " + criteria); if (!args || args.length !== 1) - throw new Error('Incorrect number of arguments for search option: ' - + criteria); - searchargs += modifier + criteria + ' ' + args[0]; - break; + throw new Error( + "Incorrect number of arguments for search option: " + criteria + ); + searchargs += modifier + criteria + " " + args[0]; + break; default: // last hope it's a seqno set // http://tools.ietf.org/html/rfc3501#section-6.4.4 - var seqnos = (args ? [criteria].concat(args) : [criteria]); + var seqnos = args ? [criteria].concat(args) : [criteria]; if (!validateUIDList(seqnos, true)) { if (seqnos.length === 0) - throw new Error('Empty sequence number list'); - searchargs += modifier + seqnos.join(','); - } else - throw new Error('Unexpected search option: ' + criteria); + throw new Error("Empty sequence number list"); + searchargs += modifier + seqnos.join(","); + } else throw new Error("Unexpected search option: " + criteria); } } - if (isOrChild) - break; + if (isOrChild) break; } return searchargs; } @@ -2038,7 +2089,6 @@ function _deepEqual(actual, expected) { // 7.1. All identical values are equivalent, as determined by ===. if (actual === expected) { return true; - } else if (Buffer.isBuffer(actual) && Buffer.isBuffer(expected)) { if (actual.length !== expected.length) return false; @@ -2048,32 +2098,34 @@ function _deepEqual(actual, expected) { return true; - // 7.2. If the expected value is a Date object, the actual value is - // equivalent if it is also a Date object that refers to the same time. + // 7.2. If the expected value is a Date object, the actual value is + // equivalent if it is also a Date object that refers to the same time. } else if (actual instanceof Date && expected instanceof Date) { return actual.getTime() === expected.getTime(); - // 7.3 If the expected value is a RegExp object, the actual value is - // equivalent if it is also a RegExp object with the same source and - // properties (`global`, `multiline`, `lastIndex`, `ignoreCase`). + // 7.3 If the expected value is a RegExp object, the actual value is + // equivalent if it is also a RegExp object with the same source and + // properties (`global`, `multiline`, `lastIndex`, `ignoreCase`). } else if (actual instanceof RegExp && expected instanceof RegExp) { - return actual.source === expected.source && - actual.global === expected.global && - actual.multiline === expected.multiline && - actual.lastIndex === expected.lastIndex && - actual.ignoreCase === expected.ignoreCase; - - // 7.4. Other pairs that do not both pass typeof value == 'object', - // equivalence is determined by ==. - } else if (typeof actual !== 'object' && typeof expected !== 'object') { + return ( + actual.source === expected.source && + actual.global === expected.global && + actual.multiline === expected.multiline && + actual.lastIndex === expected.lastIndex && + actual.ignoreCase === expected.ignoreCase + ); + + // 7.4. Other pairs that do not both pass typeof value == 'object', + // equivalence is determined by ==. + } else if (typeof actual !== "object" && typeof expected !== "object") { return actual == expected; - // 7.5 For all other Object pairs, including Array objects, equivalence is - // determined by having the same number of owned properties (as verified - // with Object.prototype.hasOwnProperty.call), the same set of keys - // (although not necessarily the same order), equivalent values for every - // corresponding key, and an identical 'prototype' property. Note: this - // accounts for both named and indexed properties on Arrays. + // 7.5 For all other Object pairs, including Array objects, equivalence is + // determined by having the same number of owned properties (as verified + // with Object.prototype.hasOwnProperty.call), the same set of keys + // (although not necessarily the same order), equivalent values for every + // corresponding key, and an identical 'prototype' property. Note: this + // accounts for both named and indexed properties on Arrays. } else { return objEquiv(actual, expected); } @@ -2082,12 +2134,11 @@ function isUndefinedOrNull(value) { return value === null || value === undefined; } function isArguments(object) { - return Object.prototype.toString.call(object) === '[object Arguments]'; + return Object.prototype.toString.call(object) === "[object Arguments]"; } function objEquiv(a, b) { var ka, kb, key, i; - if (isUndefinedOrNull(a) || isUndefinedOrNull(b)) - return false; + if (isUndefinedOrNull(a) || isUndefinedOrNull(b)) return false; // an identical 'prototype' property. if (a.prototype !== b.prototype) return false; //~~~I've managed to break Object.keys through screwy arguments passing. @@ -2103,20 +2154,19 @@ function objEquiv(a, b) { try { ka = Object.keys(a); kb = Object.keys(b); - } catch (e) {//happens when one is a string literal and the other isn't + } catch (e) { + //happens when one is a string literal and the other isn't return false; } // having the same number of owned properties (keys incorporates // hasOwnProperty) - if (ka.length !== kb.length) - return false; + if (ka.length !== kb.length) return false; //the same set of keys (although not necessarily the same order), ka.sort(); kb.sort(); //~~~cheap key test for (i = ka.length - 1; i >= 0; i--) { - if (ka[i] != kb[i]) - return false; + if (ka[i] != kb[i]) return false; } //equivalent values for every corresponding key, and //~~~possibly expensive deep test