2017-03-06 05:45:50 +08:00
|
|
|
'use strict';
|
|
|
|
|
2017-03-21 06:07:23 +08:00
|
|
|
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':
|
2017-06-01 21:55:57 +08:00
|
|
|
if (node.parsedHeader['content-type'].subtype === 'rfc822' && node.message && !options.attachmentRFC822) {
|
|
|
|
return this.processRFC822Node(node, options);
|
2017-03-06 05:45:50 +08:00
|
|
|
}
|
2017-06-01 21:55:57 +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) {
|
2017-06-01 21:55:57 +08:00
|
|
|
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';
|
|
|
|
|
2017-12-04 22:52:20 +08:00
|
|
|
if (!bodyType || !bodySubtype) {
|
|
|
|
// prevent strange content types like (NIL "/ms-word") that may break some clients
|
|
|
|
if (bodyType === 'text' || bodySubtype === 'plain') {
|
|
|
|
bodyType = 'text';
|
|
|
|
bodySubtype = 'plain';
|
|
|
|
} else {
|
|
|
|
bodyType = 'application';
|
|
|
|
bodySubtype = 'octet-stream';
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2017-03-06 05:45:50 +08:00
|
|
|
return [
|
|
|
|
// body type
|
2017-06-01 21:55:57 +08:00
|
|
|
options.upperCaseKeys ? (bodyType && bodyType.toUpperCase()) || null : bodyType,
|
2017-03-06 05:45:50 +08:00
|
|
|
|
|
|
|
// body subtype
|
2017-06-01 21:55:57 +08:00
|
|
|
options.upperCaseKeys ? (bodySubtype && bodySubtype.toUpperCase()) || null : bodySubtype,
|
2017-03-06 05:45:50 +08:00
|
|
|
|
|
|
|
// body parameter parenthesized list
|
2017-06-01 21:55:57 +08:00
|
|
|
(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
|
2017-06-01 21:55:57 +08:00
|
|
|
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 || {};
|
|
|
|
|
2017-06-01 21:55:57 +08:00
|
|
|
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
|
2017-06-01 21:55:57 +08:00
|
|
|
(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 || {};
|
|
|
|
|
2017-06-01 21:55:57 +08:00
|
|
|
let data = ((node.childNodes && node.childNodes.map(tree => this.createBodystructure(tree, options))) || [[]]).concat([
|
2017-03-06 05:45:50 +08:00
|
|
|
// body subtype
|
2017-06-01 21:55:57 +08:00
|
|
|
options.upperCaseKeys ? (node.multipart && node.multipart.toUpperCase()) || null : node.multipart,
|
2017-03-06 05:45:50 +08:00
|
|
|
|
|
|
|
// body parameter parenthesized list
|
2017-06-01 21:55:57 +08:00
|
|
|
(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 {
|
2017-07-25 21:13:10 +08:00
|
|
|
let resp = data
|
|
|
|
// skip body MD5 from extension fields
|
|
|
|
.concat(this.getExtensionFields(node, options).slice(1));
|
|
|
|
return resp;
|
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 || {};
|
|
|
|
|
2017-06-01 21:55:57 +08:00
|
|
|
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));
|
2017-06-01 21:55:57 +08:00
|
|
|
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;
|