wildduck/imap-core/lib/indexer/body-structure.js

286 lines
11 KiB
JavaScript
Raw Normal View History

2017-03-06 05:45:50 +08:00
'use strict';
const libmime = require('libmime');
const createEnvelope = require('./create-envelope');
2017-03-06 05:45:50 +08:00
class BodyStructure {
constructor(tree, options) {
this.tree = tree;
this.options = options || {};
this.currentPath = '';
this.bodyStructure = this.createBodystructure(this.tree, this.options);
}
create() {
return this.bodyStructure;
}
/**
* Generates an object out of parsed mime tree, that can be
* serialized into a BODYSTRUCTURE string
*
* @param {Object} tree Parsed mime tree (see mimeparser.js for input)
* @param {Object} [options] Optional options object
* @param {Boolean} [options.contentLanguageString] If true, convert single element array to string for Content-Language
* @param {Boolean} [options.upperCaseKeys] If true, use only upper case key names
* @param {Boolean} [options.skipContentLocation] If true, do not include Content-Location in the output
* @param {Boolean} [options.body] If true, skip extension fields (needed for BODY)
* @param {Object} Object structure in the form of BODYSTRUCTURE
*/
createBodystructure(tree, options) {
options = options || {};
let walker = node => {
switch ((node.parsedHeader['content-type'] || {}).type) {
case 'multipart':
return this.processMultipartNode(node, options);
case 'text':
return this.processTextNode(node, options);
case 'message':
if (node.parsedHeader['content-type'].subtype === 'rfc822' && node.message && !options.attachmentRFC822) {
return this.processRFC822Node(node, options);
2017-03-06 05:45:50 +08:00
}
// fall through
2017-03-06 05:45:50 +08:00
default:
return this.processAttachmentNode(node, options);
}
};
return walker(tree);
}
/**
* Generates a list of basic fields any non-multipart part should have
*
* @param {Object} node A tree node of the parsed mime tree
* @param {Object} [options] Optional options object (see createBodystructure for details)
* @return {Array} A list of basic fields
*/
getBasicFields(node, options) {
let bodyType = (node.parsedHeader['content-type'] && node.parsedHeader['content-type'].type) || null;
let bodySubtype = (node.parsedHeader['content-type'] && node.parsedHeader['content-type'].subtype) || null;
2017-03-06 05:45:50 +08:00
let contentTransfer = node.parsedHeader['content-transfer-encoding'] || '7bit';
return [
// body type
options.upperCaseKeys ? (bodyType && bodyType.toUpperCase()) || null : bodyType,
2017-03-06 05:45:50 +08:00
// body subtype
options.upperCaseKeys ? (bodySubtype && bodySubtype.toUpperCase()) || null : bodySubtype,
2017-03-06 05:45:50 +08:00
// body parameter parenthesized list
(node.parsedHeader['content-type'] &&
node.parsedHeader['content-type'].hasParams &&
this.flatten(
Object.keys(node.parsedHeader['content-type'].params).map(key => {
let value = node.parsedHeader['content-type'].params[key];
try {
value = Buffer.from(libmime.decodeWords(value).trim());
} catch (E) {
// failed to parse value
}
return [options.upperCaseKeys ? key.toUpperCase() : key, value];
})
)) ||
null,
2017-03-06 05:45:50 +08:00
// body id
node.parsedHeader['content-id'] || null,
// body description
node.parsedHeader['content-description'] || null,
// body encoding
options.upperCaseKeys ? (contentTransfer && contentTransfer.toUpperCase()) || '7bit' : contentTransfer,
2017-03-06 05:45:50 +08:00
// body size
node.size
];
}
/**
* Generates a list of extension fields any non-multipart part should have
*
* @param {Object} node A tree node of the parsed mime tree
* @param {Object} [options] Optional options object (see createBodystructure for details)
* @return {Array} A list of extension fields
*/
getExtensionFields(node, options) {
options = options || {};
let languageString = node.parsedHeader['content-language'] && node.parsedHeader['content-language'].replace(/[ ,]+/g, ',').replace(/^,+|,+$/g, '');
let language = (languageString && languageString.split(',')) || null;
2017-03-06 05:45:50 +08:00
let data;
// if `contentLanguageString` is true, then use a string instead of single element array
if (language && language.length === 1 && options.contentLanguageString) {
language = language[0];
}
data = [
// body MD5
node.parsedHeader['content-md5'] || null,
// body disposition
(node.parsedHeader['content-disposition'] && [
options.upperCaseKeys ? node.parsedHeader['content-disposition'].value.toUpperCase() : node.parsedHeader['content-disposition'].value,
(node.parsedHeader['content-disposition'].params &&
node.parsedHeader['content-disposition'].hasParams &&
this.flatten(
Object.keys(node.parsedHeader['content-disposition'].params).map(key => {
let value = node.parsedHeader['content-disposition'].params[key];
try {
value = Buffer.from(libmime.decodeWords(value).trim());
} catch (E) {
// failed to parse value
}
return [options.upperCaseKeys ? key.toUpperCase() : key, value];
})
)) ||
null
]) ||
null,
2017-03-06 05:45:50 +08:00
// body language
language
];
// if `skipContentLocation` is true, do not include Content-Location in output
//
// NB! RFC3501 has an errata with content-location type, it is described as
// 'A string list' (eg. an array) in RFC but the errata page states
// that it is a string (http://www.rfc-editor.org/errata_search.php?rfc=3501)
// see note for 'Section 7.4.2, page 75'
if (!options.skipContentLocation) {
// body location
data.push(node.parsedHeader['content-location'] || null);
}
return data;
}
/**
* Processes a node with content-type=multipart/*
*
* @param {Object} node A tree node of the parsed mime tree
* @param {Object} [options] Optional options object (see createBodystructure for details)
* @return {Array} BODYSTRUCTURE for a multipart part
*/
processMultipartNode(node, options) {
options = options || {};
let data = ((node.childNodes && node.childNodes.map(tree => this.createBodystructure(tree, options))) || [[]]).concat([
2017-03-06 05:45:50 +08:00
// body subtype
options.upperCaseKeys ? (node.multipart && node.multipart.toUpperCase()) || null : node.multipart,
2017-03-06 05:45:50 +08:00
// body parameter parenthesized list
(node.parsedHeader['content-type'] &&
node.parsedHeader['content-type'].hasParams &&
this.flatten(
Object.keys(node.parsedHeader['content-type'].params).map(key => {
let value = node.parsedHeader['content-type'].params[key];
try {
value = Buffer.from(libmime.decodeWords(value).trim());
} catch (E) {
// failed to parse value
}
return [options.upperCaseKeys ? key.toUpperCase() : key, value];
})
)) ||
null
2017-03-06 05:45:50 +08:00
]);
if (options.body) {
return data;
} else {
return (
data
// skip body MD5 from extension fields
.concat(this.getExtensionFields(node, options).slice(1))
);
2017-03-06 05:45:50 +08:00
}
}
/**
* Processes a node with content-type=text/*
*
* @param {Object} node A tree node of the parsed mime tree
* @param {Object} [options] Optional options object (see createBodystructure for details)
* @return {Array} BODYSTRUCTURE for a text part
*/
processTextNode(node, options) {
options = options || {};
let data = [].concat(this.getBasicFields(node, options)).concat([node.lineCount]);
2017-03-06 05:45:50 +08:00
if (!options.body) {
data = data.concat(this.getExtensionFields(node, options));
}
return data;
}
/**
* Processes a non-text, non-multipart node
*
* @param {Object} node A tree node of the parsed mime tree
* @param {Object} [options] Optional options object (see createBodystructure for details)
* @return {Array} BODYSTRUCTURE for the part
*/
processAttachmentNode(node, options) {
options = options || {};
let data = [].concat(this.getBasicFields(node, options));
if (!options.body) {
data = data.concat(this.getExtensionFields(node, options));
}
return data;
}
/**
* Processes a node with content-type=message/rfc822
*
* @param {Object} node A tree node of the parsed mime tree
* @param {Object} [options] Optional options object (see createBodystructure for details)
* @return {Array} BODYSTRUCTURE for a text part
*/
processRFC822Node(node, options) {
options = options || {};
let data = [].concat(this.getBasicFields(node, options));
data.push(createEnvelope(node.message.parsedHeader));
data.push(this.createBodystructure(node.message, options));
data = data.concat(node.lineCount).concat(this.getExtensionFields(node, options));
2017-03-06 05:45:50 +08:00
return data;
}
/**
* Converts all sub-arrays into one level array
* flatten([1,[2,3]]) -> [1,2,3]
*
* @param {Array} arr An array with possible sub-arrays
* @return {Array} Flat array
*/
flatten(arr) {
let result = [];
if (Array.isArray(arr)) {
arr.forEach(elm => {
if (Array.isArray(elm)) {
result = result.concat(this.flatten(elm));
} else {
result.push(elm);
}
});
} else {
result.push(arr);
}
return result;
}
}
// Expose to the world
module.exports = BodyStructure;