2020-10-17 18:42:50 +02:00

195 lines
5.9 KiB
JavaScript

'use strict';
// Helper class to rewrite nodes with specific mime type
const Transform = require('stream').Transform;
const FlowedDecoder = require('./flowed-decoder');
/**
* NodeRewriter Transform stream. Updates content for all nodes with specified mime type
*
* @constructor
* @param {String} mimeType Define the Mime-Type to look for
* @param {Function} rewriteAction Function to run with the node content
*/
class NodeRewriter extends Transform {
constructor(filterFunc, rewriteAction) {
let options = {
readableObjectMode: true,
writableObjectMode: true
};
super(options);
this.filterFunc = filterFunc;
this.rewriteAction = rewriteAction;
this.decoder = false;
this.encoder = false;
this.continue = false;
}
_transform(data, encoding, callback) {
this.processIncoming(data, callback);
}
_flush(callback) {
if (this.decoder) {
// emit an empty node just in case there is pending data to end
return this.processIncoming(
{
type: 'none'
},
callback
);
}
return callback();
}
processIncoming(data, callback) {
if (this.decoder && data.type === 'body') {
// data to parse
if (!this.decoder.write(data.value)) {
return this.decoder.once('drain', callback);
} else {
return callback();
}
} else if (this.decoder && data.type !== 'body') {
// stop decoding.
// we can not process the current data chunk as we need to wait until
// the parsed data is completely processed, so we store a reference to the
// continue callback
this.continue = () => {
this.continue = false;
this.decoder = false;
this.encoder = false;
this.processIncoming(data, callback);
};
return this.decoder.end();
} else if (data.type === 'node' && this.filterFunc(data)) {
// found matching node, create new handler
this.emit('node', this.createDecodePair(data));
} else if (this.readable && data.type !== 'none') {
// we don't care about this data, just pass it over to the joiner
this.push(data);
}
callback();
}
createDecodePair(node) {
this.decoder = node.getDecoder();
if (['base64', 'quoted-printable'].includes(node.encoding)) {
this.encoder = node.getEncoder();
} else {
this.encoder = node.getEncoder('quoted-printable');
}
let lastByte = false;
let decoder = this.decoder;
let encoder = this.encoder;
let firstChunk = true;
decoder.$reading = false;
let readFromEncoder = () => {
decoder.$reading = true;
let data = encoder.read();
if (data === null) {
decoder.$reading = false;
return;
}
if (firstChunk) {
firstChunk = false;
if (this.readable) {
this.push(node);
if (node.type === 'body') {
lastByte = node.value && node.value.length && node.value[node.value.length - 1];
}
}
}
let writeMore = true;
if (this.readable) {
writeMore = this.push({
node,
type: 'body',
value: data
});
lastByte = data && data.length && data[data.length - 1];
}
if (writeMore) {
return setImmediate(readFromEncoder);
} else {
encoder.pause();
// no idea how to catch drain? use timeout for now as poor man's substitute
// this.once('drain', () => encoder.resume());
setTimeout(() => {
encoder.resume();
setImmediate(readFromEncoder);
}, 100);
}
};
encoder.on('readable', () => {
if (!decoder.$reading) {
return readFromEncoder();
}
});
encoder.on('end', () => {
if (firstChunk) {
firstChunk = false;
if (this.readable) {
this.push(node);
if (node.type === 'body') {
lastByte = node.value && node.value.length && node.value[node.value.length - 1];
}
}
}
if (lastByte !== 0x0a) {
// make sure there is a terminating line break
this.push({
node,
type: 'body',
value: Buffer.from([0x0a])
});
}
if (this.continue) {
return this.continue();
}
});
if (/^text\//.test(node.contentType) && node.flowed) {
// text/plain; format=flowed is a special case
let flowDecoder = decoder;
decoder = new FlowedDecoder({
delSp: node.delSp,
encoding: node.encoding
});
flowDecoder.on('error', err => {
decoder.emit('error', err);
});
flowDecoder.pipe(decoder);
// we don't know what kind of data we are going to get, does it comply with the
// requirements of format=flowed, so we just cancel it
node.flowed = false;
node.delSp = false;
node.setContentType();
}
return {
node,
decoder,
encoder
};
}
}
module.exports = NodeRewriter;