emailjs-imap-client
Version:
JavaScript IMAP client
567 lines (447 loc) • 54.1 kB
JavaScript
"use strict";
Object.defineProperty(exports, "__esModule", {
value: true
});
exports.parseNAMESPACE = parseNAMESPACE;
exports.parseNAMESPACEElement = parseNAMESPACEElement;
exports.parseSELECT = parseSELECT;
exports.parseENVELOPE = parseENVELOPE;
exports.parseBODYSTRUCTURE = parseBODYSTRUCTURE;
exports.parseFETCH = parseFETCH;
exports.parseSEARCH = parseSEARCH;
exports.parseCOPY = parseCOPY;
exports.parseAPPEND = parseAPPEND;
var _emailjsAddressparser = _interopRequireDefault(require("emailjs-addressparser"));
var _emailjsImapHandler = require("emailjs-imap-handler");
var _ramda = require("ramda");
var _emailjsMimeCodec = require("emailjs-mime-codec");
function _interopRequireDefault(obj) { return obj && obj.__esModule ? obj : { default: obj }; }
function parseNAMESPACE(response) {
if (!response.payload || !response.payload.NAMESPACE || !response.payload.NAMESPACE.length) {
return false;
}
const attributes = [].concat(response.payload.NAMESPACE.pop().attributes || []);
if (!attributes.length) {
return false;
}
return {
personal: parseNAMESPACEElement(attributes[0]),
users: parseNAMESPACEElement(attributes[1]),
shared: parseNAMESPACEElement(attributes[2])
};
}
/**
* Parses a NAMESPACE element
*
* @param {Object} element
* @return {Object} Namespaces element object
*/
function parseNAMESPACEElement(element) {
if (!element) {
return false;
}
element = [].concat(element || []);
return element.map(ns => {
if (!ns || !ns.length) {
return false;
}
return {
prefix: ns[0].value,
delimiter: ns[1] && ns[1].value // The delimiter can legally be NIL which maps to null
};
});
}
/**
* Parses SELECT response
*
* @param {Object} response
* @return {Object} Mailbox information object
*/
function parseSELECT(response) {
if (!response || !response.payload) {
return;
}
const mailbox = {
readOnly: response.code === 'READ-ONLY'
};
const existsResponse = response.payload.EXISTS && response.payload.EXISTS.pop();
const flagsResponse = response.payload.FLAGS && response.payload.FLAGS.pop();
const okResponse = response.payload.OK;
if (existsResponse) {
mailbox.exists = existsResponse.nr || 0;
}
if (flagsResponse && flagsResponse.attributes && flagsResponse.attributes.length) {
mailbox.flags = flagsResponse.attributes[0].map(flag => (flag.value || '').toString().trim());
}
[].concat(okResponse || []).forEach(ok => {
switch (ok && ok.code) {
case 'PERMANENTFLAGS':
mailbox.permanentFlags = [].concat(ok.permanentflags || []);
break;
case 'UIDVALIDITY':
mailbox.uidValidity = Number(ok.uidvalidity) || 0;
break;
case 'UIDNEXT':
mailbox.uidNext = Number(ok.uidnext) || 0;
break;
case 'HIGHESTMODSEQ':
mailbox.highestModseq = ok.highestmodseq || '0'; // keep 64bit uint as a string
break;
case 'NOMODSEQ':
mailbox.noModseq = true;
break;
}
});
return mailbox;
}
/**
* Parses message envelope from FETCH response. All keys in the resulting
* object are lowercase. Address fields are all arrays with {name:, address:}
* structured values. Unicode strings are automatically decoded.
*
* @param {Array} value Envelope array
* @param {Object} Envelope object
*/
function parseENVELOPE(value) {
const envelope = {};
if (value[0] && value[0].value) {
envelope.date = value[0].value;
}
if (value[1] && value[1].value) {
envelope.subject = (0, _emailjsMimeCodec.mimeWordsDecode)(value[1] && value[1].value);
}
if (value[2] && value[2].length) {
envelope.from = processAddresses(value[2]);
}
if (value[3] && value[3].length) {
envelope.sender = processAddresses(value[3]);
}
if (value[4] && value[4].length) {
envelope['reply-to'] = processAddresses(value[4]);
}
if (value[5] && value[5].length) {
envelope.to = processAddresses(value[5]);
}
if (value[6] && value[6].length) {
envelope.cc = processAddresses(value[6]);
}
if (value[7] && value[7].length) {
envelope.bcc = processAddresses(value[7]);
}
if (value[8] && value[8].value) {
envelope['in-reply-to'] = value[8].value;
}
if (value[9] && value[9].value) {
envelope['message-id'] = value[9].value;
}
return envelope;
}
/*
* ENVELOPE lists addresses as [name-part, source-route, username, hostname]
* where source-route is not used anymore and can be ignored.
* To get comparable results with other parts of the email.js stack
* browserbox feeds the parsed address values from ENVELOPE
* to addressparser and uses resulting values instead of the
* pre-parsed addresses
*/
function processAddresses(list = []) {
return list.map(addr => {
const name = (0, _ramda.pathOr)('', ['0', 'value'], addr).trim();
const address = (0, _ramda.pathOr)('', ['2', 'value'], addr) + '@' + (0, _ramda.pathOr)('', ['3', 'value'], addr);
const formatted = name ? encodeAddressName(name) + ' <' + address + '>' : address;
const parsed = (0, _emailjsAddressparser.default)(formatted).shift(); // there should be just a single address
parsed.name = (0, _emailjsMimeCodec.mimeWordsDecode)(parsed.name);
return parsed;
});
}
/**
* If needed, encloses with quotes or mime encodes the name part of an e-mail address
*
* @param {String} name Name part of an address
* @returns {String} Mime word encoded or quoted string
*/
function encodeAddressName(name) {
if (!/^[\w ']*$/.test(name)) {
if (/^[\x20-\x7e]*$/.test(name)) {
return JSON.stringify(name);
} else {
return (0, _emailjsMimeCodec.mimeWordEncode)(name, 'Q', 52);
}
}
return name;
}
/**
* Parses message body structure from FETCH response.
*
* @param {Array} value BODYSTRUCTURE array
* @param {Object} Envelope object
*/
function parseBODYSTRUCTURE(node, path = []) {
const curNode = {};
let i = 0;
let part = 0;
if (path.length) {
curNode.part = path.join('.');
} // multipart
if (Array.isArray(node[0])) {
curNode.childNodes = [];
while (Array.isArray(node[i])) {
curNode.childNodes.push(parseBODYSTRUCTURE(node[i], path.concat(++part)));
i++;
} // multipart type
curNode.type = 'multipart/' + ((node[i++] || {}).value || '').toString().toLowerCase(); // extension data (not available for BODY requests)
// body parameter parenthesized list
if (i < node.length - 1) {
if (node[i]) {
curNode.parameters = attributesToObject(node[i]);
}
i++;
}
} else {
// content type
curNode.type = [((node[i++] || {}).value || '').toString().toLowerCase(), ((node[i++] || {}).value || '').toString().toLowerCase()].join('/'); // body parameter parenthesized list
if (node[i]) {
curNode.parameters = attributesToObject(node[i]);
}
i++; // id
if (node[i]) {
curNode.id = ((node[i] || {}).value || '').toString();
}
i++; // description
if (node[i]) {
curNode.description = ((node[i] || {}).value || '').toString();
}
i++; // encoding
if (node[i]) {
curNode.encoding = ((node[i] || {}).value || '').toString().toLowerCase();
}
i++; // size
if (node[i]) {
curNode.size = Number((node[i] || {}).value || 0) || 0;
}
i++;
if (curNode.type === 'message/rfc822') {
// message/rfc adds additional envelope, bodystructure and line count values
// envelope
if (node[i]) {
curNode.envelope = parseENVELOPE([].concat(node[i] || []));
}
i++;
if (node[i]) {
curNode.childNodes = [// rfc822 bodyparts share the same path, difference is between MIME and HEADER
// path.MIME returns message/rfc822 header
// path.HEADER returns inlined message header
parseBODYSTRUCTURE(node[i], path)];
}
i++; // line count
if (node[i]) {
curNode.lineCount = Number((node[i] || {}).value || 0) || 0;
}
i++;
} else if (/^text\//.test(curNode.type)) {
// text/* adds additional line count values
// line count
if (node[i]) {
curNode.lineCount = Number((node[i] || {}).value || 0) || 0;
}
i++;
} // extension data (not available for BODY requests)
// md5
if (i < node.length - 1) {
if (node[i]) {
curNode.md5 = ((node[i] || {}).value || '').toString().toLowerCase();
}
i++;
}
} // the following are shared extension values (for both multipart and non-multipart parts)
// not available for BODY requests
// body disposition
if (i < node.length - 1) {
if (Array.isArray(node[i]) && node[i].length) {
curNode.disposition = ((node[i][0] || {}).value || '').toString().toLowerCase();
if (Array.isArray(node[i][1])) {
curNode.dispositionParameters = attributesToObject(node[i][1]);
}
}
i++;
} // body language
if (i < node.length - 1) {
if (node[i]) {
curNode.language = [].concat(node[i]).map(val => (0, _ramda.propOr)('', 'value', val).toLowerCase());
}
i++;
} // body location
// NB! defined as a "string list" in RFC3501 but replaced in errata document with "string"
// Errata: http://www.rfc-editor.org/errata_search.php?rfc=3501
if (i < node.length - 1) {
if (node[i]) {
curNode.location = ((node[i] || {}).value || '').toString();
}
i++;
}
return curNode;
}
function attributesToObject(attrs = [], keyTransform = _ramda.toLower, valueTransform = _emailjsMimeCodec.mimeWordsDecode) {
const vals = attrs.map((0, _ramda.prop)('value'));
const keys = vals.filter((_, i) => i % 2 === 0).map(keyTransform);
const values = vals.filter((_, i) => i % 2 === 1).map(valueTransform);
return (0, _ramda.fromPairs)((0, _ramda.zip)(keys, values));
}
/**
* Parses FETCH response
*
* @param {Object} response
* @return {Object} Message object
*/
function parseFETCH(response) {
if (!response || !response.payload || !response.payload.FETCH || !response.payload.FETCH.length) {
return [];
}
const list = [];
const messages = {};
response.payload.FETCH.forEach(item => {
const params = [].concat([].concat(item.attributes || [])[0] || []); // ensure the first value is an array
let message;
let i, len, key;
if (messages[item.nr]) {
// same sequence number is already used, so merge values instead of creating a new message object
message = messages[item.nr];
} else {
messages[item.nr] = message = {
'#': item.nr
};
list.push(message);
}
for (i = 0, len = params.length; i < len; i++) {
if (i % 2 === 0) {
key = (0, _emailjsImapHandler.compiler)({
attributes: [params[i]]
}).toLowerCase().replace(/<\d+>$/, '');
continue;
}
message[key] = parseFetchValue(key, params[i]);
}
});
return list;
}
/**
* Parses a single value from the FETCH response object
*
* @param {String} key Key name (uppercase)
* @param {Mized} value Value for the key
* @return {Mixed} Processed value
*/
function parseFetchValue(key, value) {
if (!value) {
return null;
}
if (!Array.isArray(value)) {
switch (key) {
case 'uid':
case 'rfc822.size':
return Number(value.value) || 0;
case 'modseq':
// do not cast 64 bit uint to a number
return value.value || '0';
}
return value.value;
}
switch (key) {
case 'flags':
case 'x-gm-labels':
value = [].concat(value).map(flag => flag.value || '');
break;
case 'envelope':
value = parseENVELOPE([].concat(value || []));
break;
case 'bodystructure':
value = parseBODYSTRUCTURE([].concat(value || []));
break;
case 'modseq':
value = (value.shift() || {}).value || '0';
break;
}
return value;
}
/**
* Binary Search - from npm module binary-search, license CC0
*
* @param {Array} haystack Ordered array
* @param {any} needle Item to search for in haystack
* @param {Function} comparator Function that defines the sort order
* @return {Number} Index of needle in haystack or if not found,
* -Index-1 is the position where needle could be inserted while still
* keeping haystack ordered.
*/
function binSearch(haystack, needle, comparator = (a, b) => a - b) {
var mid, cmp;
var low = 0;
var high = haystack.length - 1;
while (low <= high) {
// Note that "(low + high) >>> 1" may overflow, and results in
// a typecast to double (which gives the wrong results).
mid = low + (high - low >> 1);
cmp = +comparator(haystack[mid], needle);
if (cmp < 0.0) {
// too low
low = mid + 1;
} else if (cmp > 0.0) {
// too high
high = mid - 1;
} else {
// key found
return mid;
}
} // key not found
return ~low;
}
;
/**
* Parses SEARCH response. Gathers all untagged SEARCH responses, fetched seq./uid numbers
* and compiles these into a sorted array.
*
* @param {Object} response
* @return {Object} Message object
* @param {Array} Sorted Seq./UID number list
*/
function parseSEARCH(response) {
const list = [];
if (!response || !response.payload || !response.payload.SEARCH || !response.payload.SEARCH.length) {
return list;
}
response.payload.SEARCH.forEach(result => (result.attributes || []).forEach(nr => {
nr = Number(nr && nr.value || nr) || 0;
const idx = binSearch(list, nr);
if (idx < 0) {
list.splice(-idx - 1, 0, nr);
}
}));
return list;
}
;
/**
* Parses COPY and UID COPY response.
* https://tools.ietf.org/html/rfc4315
* @param {Object} response
* @returns {{destSeqSet: string, srcSeqSet: string}} Source and
* destination uid sets if available, undefined if not.
*/
function parseCOPY(response) {
const copyuid = response && response.copyuid;
if (copyuid) {
return {
srcSeqSet: copyuid[1],
destSeqSet: copyuid[2]
};
}
}
/**
* Parses APPEND (upload) response.
* https://tools.ietf.org/html/rfc4315
* @param {Object} response
* @returns {String} The uid assigned to the uploaded message if available.
*/
function parseAPPEND(response) {
return response && response.appenduid && response.appenduid[1];
}
//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["../src/command-parser.js"],"names":["parseNAMESPACE","response","payload","NAMESPACE","length","attributes","concat","pop","personal","parseNAMESPACEElement","users","shared","element","map","ns","prefix","value","delimiter","parseSELECT","mailbox","readOnly","code","existsResponse","EXISTS","flagsResponse","FLAGS","okResponse","OK","exists","nr","flags","flag","toString","trim","forEach","ok","permanentFlags","permanentflags","uidValidity","Number","uidvalidity","uidNext","uidnext","highestModseq","highestmodseq","noModseq","parseENVELOPE","envelope","date","subject","from","processAddresses","sender","to","cc","bcc","list","addr","name","address","formatted","encodeAddressName","parsed","shift","test","JSON","stringify","parseBODYSTRUCTURE","node","path","curNode","i","part","join","Array","isArray","childNodes","push","type","toLowerCase","parameters","attributesToObject","id","description","encoding","size","lineCount","md5","disposition","dispositionParameters","language","val","location","attrs","keyTransform","toLower","valueTransform","mimeWordsDecode","vals","keys","filter","_","values","parseFETCH","FETCH","messages","item","params","message","len","key","replace","parseFetchValue","binSearch","haystack","needle","comparator","a","b","mid","cmp","low","high","parseSEARCH","SEARCH","result","idx","splice","parseCOPY","copyuid","srcSeqSet","destSeqSet","parseAPPEND","appenduid"],"mappings":"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","sourcesContent":["import parseAddress from 'emailjs-addressparser'\nimport { compiler } from 'emailjs-imap-handler'\nimport { zip, fromPairs, prop, pathOr, propOr, toLower } from 'ramda'\nimport { mimeWordEncode, mimeWordsDecode } from 'emailjs-mime-codec'\n\n/**\n * Parses NAMESPACE response\n *\n * @param {Object} response\n * @return {Object} Namespaces object\n */\nexport function parseNAMESPACE (response) {\n  if (!response.payload || !response.payload.NAMESPACE || !response.payload.NAMESPACE.length) {\n    return false\n  }\n\n  const attributes = [].concat(response.payload.NAMESPACE.pop().attributes || [])\n  if (!attributes.length) {\n    return false\n  }\n\n  return {\n    personal: parseNAMESPACEElement(attributes[0]),\n    users: parseNAMESPACEElement(attributes[1]),\n    shared: parseNAMESPACEElement(attributes[2])\n  }\n}\n\n/**\n * Parses a NAMESPACE element\n *\n * @param {Object} element\n * @return {Object} Namespaces element object\n */\nexport function parseNAMESPACEElement (element) {\n  if (!element) {\n    return false\n  }\n\n  element = [].concat(element || [])\n  return element.map((ns) => {\n    if (!ns || !ns.length) {\n      return false\n    }\n\n    return {\n      prefix: ns[0].value,\n      delimiter: ns[1] && ns[1].value // The delimiter can legally be NIL which maps to null\n    }\n  })\n}\n\n/**\n * Parses SELECT response\n *\n * @param {Object} response\n * @return {Object} Mailbox information object\n */\nexport function parseSELECT (response) {\n  if (!response || !response.payload) {\n    return\n  }\n\n  const mailbox = {\n    readOnly: response.code === 'READ-ONLY'\n  }\n  const existsResponse = response.payload.EXISTS && response.payload.EXISTS.pop()\n  const flagsResponse = response.payload.FLAGS && response.payload.FLAGS.pop()\n  const okResponse = response.payload.OK\n\n  if (existsResponse) {\n    mailbox.exists = existsResponse.nr || 0\n  }\n\n  if (flagsResponse && flagsResponse.attributes && flagsResponse.attributes.length) {\n    mailbox.flags = flagsResponse.attributes[0].map((flag) => (flag.value || '').toString().trim())\n  }\n\n  [].concat(okResponse || []).forEach((ok) => {\n    switch (ok && ok.code) {\n      case 'PERMANENTFLAGS':\n        mailbox.permanentFlags = [].concat(ok.permanentflags || [])\n        break\n      case 'UIDVALIDITY':\n        mailbox.uidValidity = Number(ok.uidvalidity) || 0\n        break\n      case 'UIDNEXT':\n        mailbox.uidNext = Number(ok.uidnext) || 0\n        break\n      case 'HIGHESTMODSEQ':\n        mailbox.highestModseq = ok.highestmodseq || '0' // keep 64bit uint as a string\n        break\n      case 'NOMODSEQ':\n        mailbox.noModseq = true\n        break\n    }\n  })\n\n  return mailbox\n}\n\n/**\n * Parses message envelope from FETCH response. All keys in the resulting\n * object are lowercase. Address fields are all arrays with {name:, address:}\n * structured values. Unicode strings are automatically decoded.\n *\n * @param {Array} value Envelope array\n * @param {Object} Envelope object\n */\nexport function parseENVELOPE (value) {\n  const envelope = {}\n\n  if (value[0] && value[0].value) {\n    envelope.date = value[0].value\n  }\n\n  if (value[1] && value[1].value) {\n    envelope.subject = mimeWordsDecode(value[1] && value[1].value)\n  }\n\n  if (value[2] && value[2].length) {\n    envelope.from = processAddresses(value[2])\n  }\n\n  if (value[3] && value[3].length) {\n    envelope.sender = processAddresses(value[3])\n  }\n\n  if (value[4] && value[4].length) {\n    envelope['reply-to'] = processAddresses(value[4])\n  }\n\n  if (value[5] && value[5].length) {\n    envelope.to = processAddresses(value[5])\n  }\n\n  if (value[6] && value[6].length) {\n    envelope.cc = processAddresses(value[6])\n  }\n\n  if (value[7] && value[7].length) {\n    envelope.bcc = processAddresses(value[7])\n  }\n\n  if (value[8] && value[8].value) {\n    envelope['in-reply-to'] = value[8].value\n  }\n\n  if (value[9] && value[9].value) {\n    envelope['message-id'] = value[9].value\n  }\n\n  return envelope\n}\n\n/*\n * ENVELOPE lists addresses as [name-part, source-route, username, hostname]\n * where source-route is not used anymore and can be ignored.\n * To get comparable results with other parts of the email.js stack\n * browserbox feeds the parsed address values from ENVELOPE\n * to addressparser and uses resulting values instead of the\n * pre-parsed addresses\n */\nfunction processAddresses (list = []) {\n  return list.map((addr) => {\n    const name = (pathOr('', ['0', 'value'], addr)).trim()\n    const address = (pathOr('', ['2', 'value'], addr)) + '@' + (pathOr('', ['3', 'value'], addr))\n    const formatted = name ? (encodeAddressName(name) + ' <' + address + '>') : address\n    const parsed = parseAddress(formatted).shift() // there should be just a single address\n    parsed.name = mimeWordsDecode(parsed.name)\n    return parsed\n  })\n}\n\n/**\n * If needed, encloses with quotes or mime encodes the name part of an e-mail address\n *\n * @param {String} name Name part of an address\n * @returns {String} Mime word encoded or quoted string\n */\nfunction encodeAddressName (name) {\n  if (!/^[\\w ']*$/.test(name)) {\n    if (/^[\\x20-\\x7e]*$/.test(name)) {\n      return JSON.stringify(name)\n    } else {\n      return mimeWordEncode(name, 'Q', 52)\n    }\n  }\n  return name\n}\n\n/**\n * Parses message body structure from FETCH response.\n *\n * @param {Array} value BODYSTRUCTURE array\n * @param {Object} Envelope object\n */\nexport function parseBODYSTRUCTURE (node, path = []) {\n  const curNode = {}\n  let i = 0\n  let part = 0\n\n  if (path.length) {\n    curNode.part = path.join('.')\n  }\n\n  // multipart\n  if (Array.isArray(node[0])) {\n    curNode.childNodes = []\n    while (Array.isArray(node[i])) {\n      curNode.childNodes.push(parseBODYSTRUCTURE(node[i], path.concat(++part)))\n      i++\n    }\n\n    // multipart type\n    curNode.type = 'multipart/' + ((node[i++] || {}).value || '').toString().toLowerCase()\n\n    // extension data (not available for BODY requests)\n\n    // body parameter parenthesized list\n    if (i < node.length - 1) {\n      if (node[i]) {\n        curNode.parameters = attributesToObject(node[i])\n      }\n      i++\n    }\n  } else {\n    // content type\n    curNode.type = [\n      ((node[i++] || {}).value || '').toString().toLowerCase(), ((node[i++] || {}).value || '').toString().toLowerCase()\n    ].join('/')\n\n    // body parameter parenthesized list\n    if (node[i]) {\n      curNode.parameters = attributesToObject(node[i])\n    }\n    i++\n\n    // id\n    if (node[i]) {\n      curNode.id = ((node[i] || {}).value || '').toString()\n    }\n    i++\n\n    // description\n    if (node[i]) {\n      curNode.description = ((node[i] || {}).value || '').toString()\n    }\n    i++\n\n    // encoding\n    if (node[i]) {\n      curNode.encoding = ((node[i] || {}).value || '').toString().toLowerCase()\n    }\n    i++\n\n    // size\n    if (node[i]) {\n      curNode.size = Number((node[i] || {}).value || 0) || 0\n    }\n    i++\n\n    if (curNode.type === 'message/rfc822') {\n      // message/rfc adds additional envelope, bodystructure and line count values\n\n      // envelope\n      if (node[i]) {\n        curNode.envelope = parseENVELOPE([].concat(node[i] || []))\n      }\n      i++\n\n      if (node[i]) {\n        curNode.childNodes = [\n          // rfc822 bodyparts share the same path, difference is between MIME and HEADER\n          // path.MIME returns message/rfc822 header\n          // path.HEADER returns inlined message header\n          parseBODYSTRUCTURE(node[i], path)\n        ]\n      }\n      i++\n\n      // line count\n      if (node[i]) {\n        curNode.lineCount = Number((node[i] || {}).value || 0) || 0\n      }\n      i++\n    } else if (/^text\\//.test(curNode.type)) {\n      // text/* adds additional line count values\n\n      // line count\n      if (node[i]) {\n        curNode.lineCount = Number((node[i] || {}).value || 0) || 0\n      }\n      i++\n    }\n\n    // extension data (not available for BODY requests)\n\n    // md5\n    if (i < node.length - 1) {\n      if (node[i]) {\n        curNode.md5 = ((node[i] || {}).value || '').toString().toLowerCase()\n      }\n      i++\n    }\n  }\n\n  // the following are shared extension values (for both multipart and non-multipart parts)\n  // not available for BODY requests\n\n  // body disposition\n  if (i < node.length - 1) {\n    if (Array.isArray(node[i]) && node[i].length) {\n      curNode.disposition = ((node[i][0] || {}).value || '').toString().toLowerCase()\n      if (Array.isArray(node[i][1])) {\n        curNode.dispositionParameters = attributesToObject(node[i][1])\n      }\n    }\n    i++\n  }\n\n  // body language\n  if (i < node.length - 1) {\n    if (node[i]) {\n      curNode.language = [].concat(node[i]).map((val) => propOr('', 'value', val).toLowerCase())\n    }\n    i++\n  }\n\n  // body location\n  // NB! defined as a \"string list\" in RFC3501 but replaced in errata document with \"string\"\n  // Errata: http://www.rfc-editor.org/errata_search.php?rfc=3501\n  if (i < node.length - 1) {\n    if (node[i]) {\n      curNode.location = ((node[i] || {}).value || '').toString()\n    }\n    i++\n  }\n\n  return curNode\n}\n\nfunction attributesToObject (attrs = [], keyTransform = toLower, valueTransform = mimeWordsDecode) {\n  const vals = attrs.map(prop('value'))\n  const keys = vals.filter((_, i) => i % 2 === 0).map(keyTransform)\n  const values = vals.filter((_, i) => i % 2 === 1).map(valueTransform)\n  return fromPairs(zip(keys, values))\n}\n\n/**\n * Parses FETCH response\n *\n * @param {Object} response\n * @return {Object} Message object\n */\nexport function parseFETCH (response) {\n  if (!response || !response.payload || !response.payload.FETCH || !response.payload.FETCH.length) {\n    return []\n  }\n\n  const list = []\n  const messages = {}\n\n  response.payload.FETCH.forEach((item) => {\n    const params = [].concat([].concat(item.attributes || [])[0] || []) // ensure the first value is an array\n    let message\n    let i, len, key\n\n    if (messages[item.nr]) {\n      // same sequence number is already used, so merge values instead of creating a new message object\n      message = messages[item.nr]\n    } else {\n      messages[item.nr] = message = {\n        '#': item.nr\n      }\n      list.push(message)\n    }\n\n    for (i = 0, len = params.length; i < len; i++) {\n      if (i % 2 === 0) {\n        key = compiler({\n          attributes: [params[i]]\n        }).toLowerCase().replace(/<\\d+>$/, '')\n        continue\n      }\n      message[key] = parseFetchValue(key, params[i])\n    }\n  })\n\n  return list\n}\n\n/**\n * Parses a single value from the FETCH response object\n *\n * @param {String} key Key name (uppercase)\n * @param {Mized} value Value for the key\n * @return {Mixed} Processed value\n */\nfunction parseFetchValue (key, value) {\n  if (!value) {\n    return null\n  }\n\n  if (!Array.isArray(value)) {\n    switch (key) {\n      case 'uid':\n      case 'rfc822.size':\n        return Number(value.value) || 0\n      case 'modseq': // do not cast 64 