2017-04-12 03:50:20 +08:00
|
|
|
'use strict';
|
|
|
|
|
|
|
|
const Transform = require('stream').Transform;
|
2017-04-25 02:20:06 +08:00
|
|
|
const Headers = require('mailsplit').Headers;
|
2017-04-12 03:50:20 +08:00
|
|
|
|
|
|
|
/**
|
|
|
|
* MessageSplitter instance is a transform stream that separates message headers
|
|
|
|
* from the rest of the body. Headers are emitted with the 'headers' event. Message
|
|
|
|
* body is passed on as the resulting stream.
|
|
|
|
*/
|
|
|
|
class MessageSplitter extends Transform {
|
|
|
|
constructor(options) {
|
|
|
|
super(options);
|
|
|
|
this.lastBytes = Buffer.alloc(4);
|
|
|
|
this.headersParsed = false;
|
|
|
|
this.headerBytes = 0;
|
|
|
|
this.headerChunks = [];
|
|
|
|
this.rawHeaders = false;
|
|
|
|
this.bodySize = 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Keeps count of the last 4 bytes in order to detect line breaks on chunk boundaries
|
|
|
|
*
|
|
|
|
* @param {Buffer} data Next data chunk from the stream
|
|
|
|
*/
|
2017-04-25 02:20:06 +08:00
|
|
|
_updateLastBytes(data) {
|
2017-04-12 03:50:20 +08:00
|
|
|
let lblen = this.lastBytes.length;
|
|
|
|
let nblen = Math.min(data.length, lblen);
|
|
|
|
|
|
|
|
// shift existing bytes
|
|
|
|
for (let i = 0, len = lblen - nblen; i < len; i++) {
|
|
|
|
this.lastBytes[i] = this.lastBytes[i + nblen];
|
|
|
|
}
|
|
|
|
|
|
|
|
// add new bytes
|
|
|
|
for (let i = 1; i <= nblen; i++) {
|
|
|
|
this.lastBytes[lblen - i] = data[data.length - i];
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Finds and removes message headers from the remaining body. We want to keep
|
|
|
|
* headers separated until final delivery to be able to modify these
|
|
|
|
*
|
|
|
|
* @param {Buffer} data Next chunk of data
|
|
|
|
* @return {Boolean} Returns true if headers are already found or false otherwise
|
|
|
|
*/
|
2017-04-25 02:20:06 +08:00
|
|
|
_checkHeaders(data) {
|
2017-04-12 03:50:20 +08:00
|
|
|
if (this.headersParsed) {
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
let lblen = this.lastBytes.length;
|
|
|
|
let headerPos = 0;
|
|
|
|
this.curLinePos = 0;
|
|
|
|
for (let i = 0, len = this.lastBytes.length + data.length; i < len; i++) {
|
|
|
|
let chr;
|
|
|
|
if (i < lblen) {
|
|
|
|
chr = this.lastBytes[i];
|
|
|
|
} else {
|
|
|
|
chr = data[i - lblen];
|
|
|
|
}
|
|
|
|
if (chr === 0x0A && i) {
|
|
|
|
let pr1 = i - 1 < lblen ? this.lastBytes[i - 1] : data[i - 1 - lblen];
|
|
|
|
let pr2 = i > 1 ? (i - 2 < lblen ? this.lastBytes[i - 2] : data[i - 2 - lblen]) : false;
|
|
|
|
if (pr1 === 0x0A) {
|
|
|
|
this.headersParsed = true;
|
|
|
|
headerPos = i - lblen + 1;
|
|
|
|
this.headerBytes += headerPos;
|
|
|
|
break;
|
|
|
|
} else if (pr1 === 0x0D && pr2 === 0x0A) {
|
|
|
|
this.headersParsed = true;
|
|
|
|
headerPos = i - lblen + 1;
|
|
|
|
this.headerBytes += headerPos;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (this.headersParsed) {
|
|
|
|
this.headerChunks.push(data.slice(0, headerPos));
|
|
|
|
this.rawHeaders = Buffer.concat(this.headerChunks, this.headerBytes);
|
|
|
|
this.headerChunks = null;
|
2017-04-25 02:20:06 +08:00
|
|
|
this.headers = new Headers(this.rawHeaders);
|
2017-04-12 03:50:20 +08:00
|
|
|
this.emit('headers', this.headers);
|
|
|
|
if (data.length - 1 > headerPos) {
|
|
|
|
let chunk = data.slice(headerPos);
|
|
|
|
this.bodySize += chunk.length;
|
|
|
|
// this would be the first chunk of data sent downstream
|
|
|
|
// from now on we keep header and body separated until final delivery
|
|
|
|
setImmediate(() => this.push(chunk));
|
|
|
|
}
|
|
|
|
return false;
|
|
|
|
} else {
|
|
|
|
this.headerBytes += data.length;
|
|
|
|
this.headerChunks.push(data);
|
|
|
|
}
|
|
|
|
|
|
|
|
// store last 4 bytes to catch header break
|
2017-04-25 02:20:06 +08:00
|
|
|
this._updateLastBytes(data);
|
2017-04-12 03:50:20 +08:00
|
|
|
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
_transform(chunk, encoding, callback) {
|
|
|
|
if (!chunk || !chunk.length) {
|
|
|
|
return callback();
|
|
|
|
}
|
|
|
|
|
|
|
|
if (typeof chunk === 'string') {
|
|
|
|
chunk = new Buffer(chunk, encoding);
|
|
|
|
}
|
|
|
|
|
|
|
|
let headersFound;
|
|
|
|
|
|
|
|
try {
|
2017-04-25 02:20:06 +08:00
|
|
|
headersFound = this._checkHeaders(chunk);
|
2017-04-12 03:50:20 +08:00
|
|
|
} catch (E) {
|
|
|
|
return callback(E);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (headersFound) {
|
|
|
|
this.bodySize += chunk.length;
|
|
|
|
this.push(chunk);
|
|
|
|
}
|
|
|
|
|
|
|
|
setImmediate(callback);
|
|
|
|
}
|
|
|
|
|
|
|
|
_flush(callback) {
|
|
|
|
if (this.headerChunks) {
|
|
|
|
// all chunks are checked but we did not find where the body starts
|
|
|
|
// so emit all we got as headers and push empty line as body
|
|
|
|
this.headersParsed = true;
|
|
|
|
// add header terminator
|
|
|
|
this.headerChunks.push(Buffer.from('\r\n\r\n'));
|
|
|
|
this.headerBytes += 4;
|
|
|
|
// join all chunks into a header block
|
|
|
|
this.rawHeaders = Buffer.concat(this.headerChunks, this.headerBytes);
|
|
|
|
|
2017-04-25 02:20:06 +08:00
|
|
|
this.headers = new Headers(this.rawHeaders);
|
2017-04-12 03:50:20 +08:00
|
|
|
|
|
|
|
this.emit('headers', this.headers);
|
|
|
|
this.headerChunks = null;
|
|
|
|
|
|
|
|
// this is our body
|
|
|
|
this.push(Buffer.from('\r\n'));
|
|
|
|
}
|
|
|
|
callback();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
module.exports = MessageSplitter;
|