mirror of
https://github.com/nodemailer/wildduck.git
synced 2025-01-19 14:30:52 +08:00
172 lines
5.5 KiB
JavaScript
Executable file
172 lines
5.5 KiB
JavaScript
Executable file
'use strict';
|
|
|
|
const stream = require('stream');
|
|
const Writable = stream.Writable;
|
|
const PassThrough = stream.PassThrough;
|
|
|
|
/**
|
|
* Incoming IMAP stream parser. Detects and emits command payloads.
|
|
* If literal values are encountered the command payload is split into parts
|
|
* and all parts are emitted separately. The client must send the +\r\n or
|
|
* return a NO error for the literal
|
|
*
|
|
* @constructor
|
|
* @param {Object} [options] Optional Stream options object
|
|
*/
|
|
class IMAPStream extends Writable {
|
|
constructor(options) {
|
|
// init Writable
|
|
super();
|
|
|
|
this.options = options || {};
|
|
Writable.call(this, this.options);
|
|
|
|
// unprocessed chars from the last parsing iteration
|
|
this._remainder = '';
|
|
this._literal = false;
|
|
this._literalReady = false;
|
|
|
|
// how many literal bytes to wait for
|
|
this._expecting = 0;
|
|
|
|
// once the input stream ends, flush all output without expecting the newline
|
|
this.on('finish', this._flushData.bind(this));
|
|
}
|
|
|
|
/**
|
|
* Placeholder command handler. Override this with your own.
|
|
*/
|
|
oncommand(/* command, callback */) {
|
|
throw new Error('Command handler is not set');
|
|
}
|
|
|
|
// PRIVATE METHODS
|
|
|
|
/**
|
|
* Writable._write method.
|
|
*/
|
|
_write(chunk, encoding, done) {
|
|
if (!chunk || !chunk.length) {
|
|
return done();
|
|
}
|
|
|
|
let data = this._remainder + chunk.toString('binary');
|
|
this._remainder = '';
|
|
|
|
// start reading data
|
|
// regex is passed as an argument because we need to keep count of the lastIndex property
|
|
this._readValue(/\r?\n/g, data, 0, done);
|
|
}
|
|
|
|
/**
|
|
* Reads next command from incoming stream
|
|
*
|
|
* @param {RegExp} regex Regular expression object. Needed to keep lastIndex value
|
|
* @param {String} data Incoming data as binary string
|
|
* @param {Number} pos Cursor position in current data chunk
|
|
* @param {Function} done Function to call once data is processed
|
|
*/
|
|
_readValue(regex, data, pos, done) {
|
|
let match;
|
|
let line;
|
|
|
|
// Handle literal mode where we know how many bytes to expect before switching back to
|
|
// normal line based mode. All the data we receive is pumped to a passthrough stream
|
|
if (this._expecting > 0) {
|
|
if (data.length - pos <= 0) {
|
|
return done();
|
|
}
|
|
|
|
if (data.length - pos >= this._expecting) {
|
|
// all bytes received
|
|
this._literal.end(Buffer.from(data.substr(pos, this._expecting), 'binary'));
|
|
pos += this._expecting;
|
|
this._expecting = 0;
|
|
this._literal = false;
|
|
|
|
if (this._literalReady) {
|
|
// can continue
|
|
this._literalReady = false;
|
|
} else {
|
|
this._literalReady = this._readValue.bind(this, /\r?\n/g, data.substr(pos), 0, done);
|
|
return;
|
|
}
|
|
} else {
|
|
// data still pending
|
|
this._literal.write(Buffer.from(data.substr(pos), 'binary'), done);
|
|
this._expecting -= data.length - pos;
|
|
return; // wait for the next chunk
|
|
}
|
|
}
|
|
|
|
// search for the next newline
|
|
// exec keeps count of the last match with lastIndex
|
|
// so it knows from where to start with the next iteration
|
|
if ((match = regex.exec(data))) {
|
|
line = data.substr(pos, match.index - pos);
|
|
pos += line.length + match[0].length;
|
|
} else {
|
|
this._remainder = pos < data.length ? data.substr(pos) : '';
|
|
return done();
|
|
}
|
|
|
|
if ((match = /\{(\d+)\}$/.exec(line))) {
|
|
this._expecting = Number(match[1]);
|
|
if (!isNaN(match[1])) {
|
|
this._literal = new PassThrough();
|
|
|
|
this.oncommand(
|
|
{
|
|
value: line,
|
|
final: false,
|
|
expecting: this._expecting,
|
|
literal: this._literal,
|
|
|
|
// called once the stream has been processed
|
|
readyCallback: () => {
|
|
let next = this._literalReady;
|
|
if (typeof next === 'function') {
|
|
this._literalReady = false;
|
|
next();
|
|
} else {
|
|
this._literalReady = true;
|
|
}
|
|
}
|
|
},
|
|
err => {
|
|
if (err) {
|
|
this._expecting = 0;
|
|
this._literal = false;
|
|
this._literalReady = false;
|
|
}
|
|
setImmediate(this._readValue.bind(this, regex, data, pos, done));
|
|
}
|
|
);
|
|
return;
|
|
}
|
|
}
|
|
|
|
this.oncommand(
|
|
{
|
|
value: line,
|
|
final: true
|
|
},
|
|
this._readValue.bind(this, regex, data, pos, done)
|
|
);
|
|
}
|
|
|
|
/**
|
|
* Flushes remaining bytes
|
|
*/
|
|
_flushData() {
|
|
let line;
|
|
if (this._remainder) {
|
|
line = this._remainder;
|
|
this._remainder = '';
|
|
this.oncommand(Buffer.from(line, 'binary'));
|
|
}
|
|
}
|
|
}
|
|
|
|
// Expose to the world
|
|
module.exports.IMAPStream = IMAPStream;
|