115 lines
292 KiB
JavaScript
115 lines
292 KiB
JavaScript
"use strict";
|
|
/*
|
|
* ATTENTION: An "eval-source-map" devtool has been used.
|
|
* This devtool is neither made for production nor for readable output files.
|
|
* It uses "eval()" calls to create a separate source file with attached SourceMaps in the browser devtools.
|
|
* If you are trying to read the output file, select a different devtool (https://webpack.js.org/configuration/devtool/)
|
|
* or disable the default devtool with "devtool: false".
|
|
* If you are looking for production-ready output files, see mode: "production" (https://webpack.js.org/configuration/mode/).
|
|
*/
|
|
exports.id = "vendor-chunks/mailparser";
|
|
exports.ids = ["vendor-chunks/mailparser"];
|
|
exports.modules = {
|
|
|
|
/***/ "(action-browser)/./node_modules/mailparser/index.js":
|
|
/*!******************************************!*\
|
|
!*** ./node_modules/mailparser/index.js ***!
|
|
\******************************************/
|
|
/***/ ((module, __unused_webpack_exports, __webpack_require__) => {
|
|
|
|
eval("\n\nconst MailParser = __webpack_require__(/*! ./lib/mail-parser */ \"(action-browser)/./node_modules/mailparser/lib/mail-parser.js\");\nconst simpleParser = __webpack_require__(/*! ./lib/simple-parser */ \"(action-browser)/./node_modules/mailparser/lib/simple-parser.js\");\n\nmodule.exports = {\n MailParser,\n simpleParser\n};\n//# sourceURL=[module]\n//# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiKGFjdGlvbi1icm93c2VyKS8uL25vZGVfbW9kdWxlcy9tYWlscGFyc2VyL2luZGV4LmpzIiwibWFwcGluZ3MiOiJBQUFhOztBQUViLG1CQUFtQixtQkFBTyxDQUFDLHdGQUFtQjtBQUM5QyxxQkFBcUIsbUJBQU8sQ0FBQyw0RkFBcUI7O0FBRWxEO0FBQ0E7QUFDQTtBQUNBIiwic291cmNlcyI6WyIvaG9tZS9hbG1hL25leHRnZW4vTmVhaC1tYWlsL25vZGVfbW9kdWxlcy9tYWlscGFyc2VyL2luZGV4LmpzIl0sInNvdXJjZXNDb250ZW50IjpbIid1c2Ugc3RyaWN0JztcblxuY29uc3QgTWFpbFBhcnNlciA9IHJlcXVpcmUoJy4vbGliL21haWwtcGFyc2VyJyk7XG5jb25zdCBzaW1wbGVQYXJzZXIgPSByZXF1aXJlKCcuL2xpYi9zaW1wbGUtcGFyc2VyJyk7XG5cbm1vZHVsZS5leHBvcnRzID0ge1xuICAgIE1haWxQYXJzZXIsXG4gICAgc2ltcGxlUGFyc2VyXG59O1xuIl0sIm5hbWVzIjpbXSwiaWdub3JlTGlzdCI6WzBdLCJzb3VyY2VSb290IjoiIn0=\n//# sourceURL=webpack-internal:///(action-browser)/./node_modules/mailparser/index.js\n");
|
|
|
|
/***/ }),
|
|
|
|
/***/ "(action-browser)/./node_modules/mailparser/lib/mail-parser.js":
|
|
/*!****************************************************!*\
|
|
!*** ./node_modules/mailparser/lib/mail-parser.js ***!
|
|
\****************************************************/
|
|
/***/ ((module, __unused_webpack_exports, __webpack_require__) => {
|
|
|
|
eval("\n\nconst mailsplit = __webpack_require__(/*! mailsplit */ \"(action-browser)/./node_modules/mailsplit/index.js\");\nconst libmime = __webpack_require__(/*! libmime */ \"(action-browser)/./node_modules/libmime/lib/libmime.js\");\nconst addressparser = __webpack_require__(/*! nodemailer/lib/addressparser */ \"(action-browser)/./node_modules/mailparser/node_modules/nodemailer/lib/addressparser/index.js\");\nconst Transform = (__webpack_require__(/*! stream */ \"stream\").Transform);\nconst Splitter = mailsplit.Splitter;\nconst punycode = __webpack_require__(/*! punycode.js */ \"(action-browser)/./node_modules/punycode.js/punycode.es6.js\");\nconst FlowedDecoder = __webpack_require__(/*! mailsplit/lib/flowed-decoder */ \"(action-browser)/./node_modules/mailsplit/lib/flowed-decoder.js\");\nconst StreamHash = __webpack_require__(/*! ./stream-hash */ \"(action-browser)/./node_modules/mailparser/lib/stream-hash.js\");\nconst iconv = __webpack_require__(/*! iconv-lite */ \"(action-browser)/./node_modules/iconv-lite/lib/index.js\");\nconst { htmlToText } = __webpack_require__(/*! html-to-text */ \"(action-browser)/./node_modules/html-to-text/lib/html-to-text.cjs\");\nconst he = __webpack_require__(/*! he */ \"(action-browser)/./node_modules/he/he.js\");\nconst linkify = __webpack_require__(/*! linkify-it */ \"(action-browser)/./node_modules/linkify-it/build/index.cjs.js\")();\nconst tlds = __webpack_require__(/*! tlds */ \"(action-browser)/./node_modules/tlds/index.json\");\nconst encodingJapanese = __webpack_require__(/*! encoding-japanese */ \"(action-browser)/./node_modules/encoding-japanese/src/index.js\");\n\nlinkify\n .tlds(tlds) // Reload with full tlds list\n .tlds('onion', true) // Add unofficial `.onion` domain\n .add('git:', 'http:') // Add `git:` ptotocol as \"alias\"\n .add('ftp:', null) // Disable `ftp:` ptotocol\n .set({ fuzzyIP: true, fuzzyLink: true, fuzzyEmail: true });\n\n// twitter linkifier from\n// https://github.com/markdown-it/linkify-it#example-2-add-twitter-mentions-handler\nlinkify.add('@', {\n validate(text, pos, self) {\n let tail = text.slice(pos);\n\n if (!self.re.twitter) {\n self.re.twitter = new RegExp('^([a-zA-Z0-9_]){1,15}(?!_)(?=$|' + self.re.src_ZPCc + ')');\n }\n if (self.re.twitter.test(tail)) {\n // Linkifier allows punctuation chars before prefix,\n // but we additionally disable `@` (\"@@mention\" is invalid)\n if (pos >= 2 && tail[pos - 2] === '@') {\n return false;\n }\n return tail.match(self.re.twitter)[0].length;\n }\n return 0;\n },\n normalize(match) {\n match.url = 'https://twitter.com/' + match.url.replace(/^@/, '');\n }\n});\n\nclass IconvDecoder extends Transform {\n constructor(Iconv, charset) {\n super();\n\n // Iconv throws error on ks_c_5601-1987 when it is mapped to EUC-KR\n // https://github.com/bnoordhuis/node-iconv/issues/169\n if (charset.toLowerCase() === 'ks_c_5601-1987') {\n charset = 'CP949';\n }\n this.stream = new Iconv(charset, 'UTF-8//TRANSLIT//IGNORE');\n\n this.inputEnded = false;\n this.endCb = false;\n\n this.stream.on('error', err => this.emit('error', err));\n this.stream.on('data', chunk => this.push(chunk));\n this.stream.on('end', () => {\n this.inputEnded = true;\n if (typeof this.endCb === 'function') {\n this.endCb();\n }\n });\n }\n\n _transform(chunk, encoding, done) {\n this.stream.write(chunk);\n done();\n }\n\n _flush(done) {\n this.endCb = done;\n this.stream.end();\n }\n}\n\nclass JPDecoder extends Transform {\n constructor(charset) {\n super();\n\n this.charset = charset;\n this.chunks = [];\n this.chunklen = 0;\n }\n\n _transform(chunk, encoding, done) {\n if (typeof chunk === 'string') {\n chunk = Buffer.from(chunk, encoding);\n }\n\n this.chunks.push(chunk);\n this.chunklen += chunk.length;\n done();\n }\n\n _flush(done) {\n let input = Buffer.concat(this.chunks, this.chunklen);\n try {\n let output = encodingJapanese.convert(input, {\n to: 'UNICODE', // to_encoding\n from: this.charset, // from_encoding\n type: 'string'\n });\n if (typeof output === 'string') {\n output = Buffer.from(output);\n }\n this.push(output);\n } catch (err) {\n // keep as is on errors\n this.push(input);\n }\n\n done();\n }\n}\n\nclass MailParser extends Transform {\n constructor(config) {\n super({\n readableObjectMode: true,\n writableObjectMode: false\n });\n\n this.options = config || {};\n this.splitter = new Splitter(config);\n this.finished = false;\n this.waitingEnd = false;\n\n this.headers = false;\n this.headerLines = false;\n\n this.endReceived = false;\n this.reading = false;\n this.hasFailed = false;\n\n this.tree = false;\n this.curnode = false;\n this.waitUntilAttachmentEnd = false;\n this.attachmentCallback = false;\n\n this.hasHtml = false;\n this.hasText = false;\n\n this.text = false;\n this.html = false;\n this.textAsHtml = false;\n\n this.attachmentList = [];\n\n this.boundaries = [];\n\n this.textTypes = ['text/plain', 'text/html'].concat(!this.options.keepDeliveryStatus ? 'message/delivery-status' : []);\n\n this.decoder = this.getDecoder();\n\n this.splitter.on('readable', () => {\n if (this.reading) {\n return false;\n }\n this.readData();\n });\n\n this.splitter.on('end', () => {\n this.endReceived = true;\n if (!this.reading) {\n this.endStream();\n }\n });\n\n this.splitter.on('error', err => {\n this.hasFailed = true;\n if (typeof this.waitingEnd === 'function') {\n return this.waitingEnd(err);\n }\n this.emit('error', err);\n });\n\n this.libmime = new libmime.Libmime({ Iconv: this.options.Iconv });\n }\n\n getDecoder() {\n if (this.options.Iconv) {\n const Iconv = this.options.Iconv;\n // create wrapper\n return {\n decodeStream(charset) {\n return new IconvDecoder(Iconv, charset);\n }\n };\n } else {\n return {\n decodeStream(charset) {\n charset = (charset || 'ascii').toString().trim().toLowerCase();\n if (/^jis|^iso-?2022-?jp|^EUCJP/i.test(charset)) {\n // special case not supported by iconv-lite\n return new JPDecoder(charset);\n }\n\n return iconv.decodeStream(charset);\n }\n };\n }\n }\n\n readData() {\n if (this.hasFailed) {\n return false;\n }\n this.reading = true;\n let data = this.splitter.read();\n if (data === null) {\n this.reading = false;\n if (this.endReceived) {\n this.endStream();\n }\n return;\n }\n\n this.processChunk(data, err => {\n if (err) {\n if (typeof this.waitingEnd === 'function') {\n return this.waitingEnd(err);\n }\n return this.emit('error', err);\n }\n setImmediate(() => this.readData());\n });\n }\n\n endStream() {\n this.finished = true;\n\n if (this.curnode && this.curnode.decoder) {\n this.curnode.decoder.end();\n }\n if (typeof this.waitingEnd === 'function') {\n this.waitingEnd();\n }\n }\n\n _transform(chunk, encoding, done) {\n if (!chunk || !chunk.length) {\n return done();\n }\n\n if (this.splitter.write(chunk) === false) {\n return this.splitter.once('drain', () => {\n done();\n });\n } else {\n return done();\n }\n }\n\n _flush(done) {\n setImmediate(() => this.splitter.end());\n if (this.finished) {\n return this.cleanup(done);\n }\n this.waitingEnd = () => {\n this.cleanup(() => {\n done();\n });\n };\n }\n\n cleanup(done) {\n let finish = () => {\n try {\n let t = this.getTextContent();\n this.push(t);\n } catch (err) {\n return this.emit('error', err);\n }\n\n done();\n };\n\n if (this.curnode && this.curnode.decoder && this.curnode.decoder.readable && !this.decoderEnded) {\n (this.curnode.contentStream || this.curnode.decoder).once('end', () => {\n finish();\n });\n this.curnode.decoder.end();\n } else {\n setImmediate(() => {\n finish();\n });\n }\n }\n\n processHeaders(lines) {\n let headers = new Map();\n (lines || []).forEach(line => {\n let key = line.key;\n let value = ((this.libmime.decodeHeader(line.line) || {}).value || '').toString().trim();\n value = Buffer.from(value, 'binary').toString();\n switch (key) {\n case 'content-type':\n case 'content-disposition':\n case 'dkim-signature':\n value = this.libmime.parseHeaderValue(value);\n if (value.value) {\n value.value = this.libmime.decodeWords(value.value);\n }\n Object.keys((value && value.params) || {}).forEach(key => {\n try {\n value.params[key] = this.libmime.decodeWords(value.params[key]);\n } catch (E) {\n // ignore, keep as is\n }\n });\n break;\n case 'date': {\n let dateValue = new Date(value);\n if (isNaN(dateValue)) {\n // date parsing failed :S\n dateValue = new Date();\n }\n value = dateValue;\n break;\n }\n case 'subject':\n try {\n value = this.libmime.decodeWords(value);\n } catch (E) {\n // ignore, keep as is\n }\n break;\n case 'references':\n try {\n value = this.libmime.decodeWords(value);\n } catch (E) {\n // ignore\n }\n value = value.split(/\\s+/).map(this.ensureMessageIDFormat);\n break;\n case 'message-id':\n case 'in-reply-to':\n try {\n value = this.libmime.decodeWords(value);\n } catch (E) {\n // ignore\n }\n value = this.ensureMessageIDFormat(value);\n break;\n case 'priority':\n case 'x-priority':\n case 'x-msmail-priority':\n case 'importance':\n key = 'priority';\n value = this.parsePriority(value);\n break;\n case 'from':\n case 'to':\n case 'cc':\n case 'bcc':\n case 'sender':\n case 'reply-to':\n case 'delivered-to':\n case 'return-path':\n case 'disposition-notification-to':\n value = addressparser(value);\n this.decodeAddresses(value);\n value = {\n value,\n html: this.getAddressesHTML(value),\n text: this.getAddressesText(value)\n };\n break;\n }\n\n // handle list-* keys\n if (key.substr(0, 5) === 'list-') {\n value = this.parseListHeader(key.substr(5), value);\n key = 'list';\n }\n\n if (value) {\n if (!headers.has(key)) {\n headers.set(key, [].concat(value || []));\n } else if (Array.isArray(value)) {\n headers.set(key, headers.get(key).concat(value));\n } else {\n headers.get(key).push(value);\n }\n }\n });\n\n // keep only the first value\n let singleKeys = [\n 'message-id',\n 'content-id',\n 'from',\n 'sender',\n 'in-reply-to',\n 'reply-to',\n 'subject',\n 'date',\n 'content-disposition',\n 'content-type',\n 'content-transfer-encoding',\n 'priority',\n 'mime-version',\n 'content-description',\n 'precedence',\n 'errors-to',\n 'disposition-notification-to'\n ];\n\n headers.forEach((value, key) => {\n if (Array.isArray(value)) {\n if (singleKeys.includes(key) && value.length) {\n headers.set(key, value[value.length - 1]);\n } else if (value.length === 1) {\n headers.set(key, value[0]);\n }\n }\n\n if (key === 'list') {\n // normalize List-* headers\n let listValue = {};\n [].concat(value || []).forEach(val => {\n Object.keys(val || {}).forEach(listKey => {\n listValue[listKey] = val[listKey];\n });\n });\n headers.set(key, listValue);\n }\n });\n\n return headers;\n }\n\n parseListHeader(key, value) {\n let addresses = addressparser(value);\n let response = {};\n let data = addresses\n .map(address => {\n if (/^https?:/i.test(address.name)) {\n response.url = address.name;\n } else if (address.name) {\n response.name = address.name;\n }\n if (/^mailto:/.test(address.address)) {\n response.mail = address.address.substr(7);\n } else if (address.address && address.address.indexOf('@') < 0) {\n response.id = address.address;\n } else if (address.address) {\n response.mail = address.address;\n }\n if (Object.keys(response).length) {\n return response;\n }\n return false;\n })\n .filter(address => address);\n if (data.length) {\n return {\n [key]: response\n };\n }\n return false;\n }\n\n parsePriority(value) {\n value = value.toLowerCase().trim();\n if (!isNaN(parseInt(value, 10))) {\n // support \"X-Priority: 1 (Highest)\"\n value = parseInt(value, 10) || 0;\n if (value === 3) {\n return 'normal';\n } else if (value > 3) {\n return 'low';\n } else {\n return 'high';\n }\n } else {\n switch (value) {\n case 'non-urgent':\n case 'low':\n return 'low';\n case 'urgent':\n case 'high':\n return 'high';\n }\n }\n return 'normal';\n }\n\n ensureMessageIDFormat(value) {\n if (!value.length) {\n return false;\n }\n\n if (value.charAt(0) !== '<') {\n value = '<' + value;\n }\n\n if (value.charAt(value.length - 1) !== '>') {\n value += '>';\n }\n\n return value;\n }\n\n decodeAddresses(addresses) {\n let processedAddress = new WeakSet();\n for (let i = 0; i < addresses.length; i++) {\n let address = addresses[i];\n address.name = (address.name || '').toString().trim();\n\n if (!address.address && /^(=\\?([^?]+)\\?[Bb]\\?[^?]*\\?=)(\\s*=\\?([^?]+)\\?[Bb]\\?[^?]*\\?=)*$/.test(address.name) && !processedAddress.has(address)) {\n let parsed = addressparser(this.libmime.decodeWords(address.name));\n if (parsed.length) {\n parsed.forEach(entry => {\n processedAddress.add(entry);\n addresses.push(entry);\n });\n }\n\n // remove current element\n addresses.splice(i, 1);\n i--;\n continue;\n }\n\n if (address.name) {\n try {\n address.name = this.libmime.decodeWords(address.name);\n } catch (E) {\n //ignore, keep as is\n }\n }\n if (/@xn--/.test(address.address)) {\n try {\n address.address =\n address.address.substr(0, address.address.lastIndexOf('@') + 1) +\n punycode.toUnicode(address.address.substr(address.address.lastIndexOf('@') + 1));\n } catch (E) {\n // Not a valid punycode string; keep as is\n }\n }\n if (address.group) {\n this.decodeAddresses(address.group);\n }\n }\n }\n\n createNode(node) {\n let contentType = node.contentType;\n let disposition = node.disposition;\n let encoding = node.encoding;\n let charset = node.charset;\n\n if (!contentType && node.root) {\n contentType = 'text/plain';\n }\n\n let newNode = {\n node,\n headerLines: node.headers.lines,\n headers: this.processHeaders(node.headers.getList()),\n contentType,\n children: []\n };\n\n if (!/^multipart\\//i.test(contentType)) {\n if (disposition && !['attachment', 'inline'].includes(disposition)) {\n disposition = 'attachment';\n }\n\n if (!disposition && !this.textTypes.includes(contentType)) {\n newNode.disposition = 'attachment';\n } else {\n newNode.disposition = disposition || 'inline';\n }\n\n newNode.isAttachment = !this.textTypes.includes(contentType) || newNode.disposition !== 'inline';\n\n newNode.encoding = ['quoted-printable', 'base64'].includes(encoding) ? encoding : 'binary';\n\n if (charset) {\n newNode.charset = charset;\n }\n\n let decoder = node.getDecoder();\n decoder.on('end', () => {\n this.decoderEnded = true;\n });\n newNode.decoder = decoder;\n }\n\n if (node.root) {\n this.headers = newNode.headers;\n this.headerLines = newNode.headerLines;\n }\n\n // find location in tree\n\n if (!this.tree) {\n newNode.root = true;\n this.curnode = this.tree = newNode;\n return newNode;\n }\n\n // immediate child of root node\n if (!this.curnode.parent) {\n newNode.parent = this.curnode;\n this.curnode.children.push(newNode);\n this.curnode = newNode;\n return newNode;\n }\n\n // siblings\n if (this.curnode.parent.node === node.parentNode) {\n newNode.parent = this.curnode.parent;\n this.curnode.parent.children.push(newNode);\n this.curnode = newNode;\n return newNode;\n }\n\n // first child\n if (this.curnode.node === node.parentNode) {\n newNode.parent = this.curnode;\n this.curnode.children.push(newNode);\n this.curnode = newNode;\n return newNode;\n }\n\n // move up\n let parentNode = this.curnode;\n while ((parentNode = parentNode.parent)) {\n if (parentNode.node === node.parentNode) {\n newNode.parent = parentNode;\n parentNode.children.push(newNode);\n this.curnode = newNode;\n return newNode;\n }\n }\n\n // should never happen, can't detect parent\n this.curnode = newNode;\n return newNode;\n }\n\n getTextContent() {\n let text = [];\n let html = [];\n let processNode = (alternative, level, node) => {\n if (node.showMeta) {\n let meta = ['From', 'Subject', 'Date', 'To', 'Cc', 'Bcc']\n .map(fkey => {\n let key = fkey.toLowerCase();\n if (!node.headers.has(key)) {\n return false;\n }\n let value = node.headers.get(key);\n if (!value) {\n return false;\n }\n return {\n key: fkey,\n value: Array.isArray(value) ? value[value.length - 1] : value\n };\n })\n .filter(entry => entry);\n if (this.hasHtml) {\n html.push(\n '<table class=\"mp_head\">' +\n meta\n .map(entry => {\n let value = entry.value;\n switch (entry.key) {\n case 'From':\n case 'To':\n case 'Cc':\n case 'Bcc':\n value = value.html;\n break;\n case 'Date':\n value = this.options.formatDateString ? this.options.formatDateString(value) : value.toUTCString();\n break;\n case 'Subject':\n value = '<strong>' + he.encode(value) + '</strong>';\n break;\n default:\n value = he.encode(value);\n }\n\n return '<tr><td class=\"mp_head_key\">' + he.encode(entry.key) + ':</td><td class=\"mp_head_value\">' + value + '<td></tr>';\n })\n .join('\\n') +\n '<table>'\n );\n }\n if (this.hasText) {\n text.push(\n '\\n' +\n meta\n .map(entry => {\n let value = entry.value;\n switch (entry.key) {\n case 'From':\n case 'To':\n case 'Cc':\n case 'Bcc':\n value = value.text;\n break;\n case 'Date':\n value = this.options.formatDateString ? this.options.formatDateString(value) : value.toUTCString();\n break;\n }\n return entry.key + ': ' + value;\n })\n .join('\\n') +\n '\\n'\n );\n }\n }\n if (node.textContent) {\n if (node.contentType === 'text/plain') {\n text.push(node.textContent);\n if (!alternative && this.hasHtml) {\n html.push(this.textToHtml(node.textContent));\n }\n } else if (node.contentType === 'message/delivery-status' && !this.options.keepDeliveryStatus) {\n text.push(node.textContent);\n if (!alternative && this.hasHtml) {\n html.push(this.textToHtml(node.textContent));\n }\n } else if (node.contentType === 'text/html') {\n let failedToParseHtml = false;\n if ((!alternative && this.hasText) || (node.root && !this.hasText)) {\n if (this.options.skipHtmlToText) {\n text.push('');\n } else if (node.textContent.length > this.options.maxHtmlLengthToParse) {\n this.emit('error', new Error(`HTML too long for parsing ${node.textContent.length} bytes`));\n text.push('Invalid HTML content (too long)');\n failedToParseHtml = true;\n } else {\n try {\n text.push(htmlToText(node.textContent));\n } catch (err) {\n this.emit('error', new Error('Failed to parse HTML'));\n text.push('Invalid HTML content');\n failedToParseHtml = true;\n }\n }\n }\n if (!failedToParseHtml) {\n html.push(node.textContent);\n }\n }\n }\n alternative = alternative || node.contentType === 'multipart/alternative';\n if (node.children) {\n node.children.forEach(subNode => {\n processNode(alternative, level + 1, subNode);\n });\n }\n };\n\n processNode(false, 0, this.tree);\n\n let response = {\n type: 'text'\n };\n if (html.length) {\n this.html = response.html = html.join('<br/>\\n');\n }\n if (text.length) {\n this.text = response.text = text.join('\\n');\n this.textAsHtml = response.textAsHtml = text.map(part => this.textToHtml(part)).join('<br/>\\n');\n }\n return response;\n }\n\n processChunk(data, done) {\n let partId = null;\n if (data._parentBoundary) {\n partId = this._getPartId(data._parentBoundary);\n }\n switch (data.type) {\n case 'node': {\n let node = this.createNode(data);\n if (node === this.tree) {\n ['subject', 'references', 'date', 'to', 'from', 'to', 'cc', 'bcc', 'message-id', 'in-reply-to', 'reply-to'].forEach(key => {\n if (node.headers.has(key)) {\n this[key.replace(/-([a-z])/g, (m, c) => c.toUpperCase())] = node.headers.get(key);\n }\n });\n this.emit('headers', node.headers);\n\n if (node.headerLines) {\n this.emit('headerLines', node.headerLines);\n } \n }\n\n if (data.contentType === 'message/rfc822' && data.messageNode) {\n break;\n }\n\n if (data.parentNode && data.parentNode.contentType === 'message/rfc822') {\n node.showMeta = true;\n }\n\n if (node.isAttachment) {\n let contentType = node.contentType;\n if (node.contentType === 'application/octet-stream' && data.filename) {\n contentType = this.libmime.detectMimeType(data.filename) || 'application/octet-stream';\n }\n\n let attachment = {\n type: 'attachment',\n content: null,\n contentType,\n partId,\n release: () => {\n attachment.release = null;\n if (this.waitUntilAttachmentEnd && typeof this.attachmentCallback === 'function') {\n setImmediate(this.attachmentCallback);\n }\n this.attachmentCallback = false;\n this.waitUntilAttachmentEnd = false;\n }\n };\n\n let algo = this.options.checksumAlgo || 'md5';\n let hasher = new StreamHash(attachment, algo);\n node.decoder.on('error', err => {\n hasher.emit('error', err);\n });\n\n node.decoder.on('readable', () => {\n let chunk;\n\n while ((chunk = node.decoder.read()) !== null) {\n hasher.write(chunk);\n }\n });\n\n node.decoder.once('end', () => {\n hasher.end();\n });\n\n //node.decoder.pipe(hasher);\n attachment.content = hasher;\n\n this.waitUntilAttachmentEnd = true;\n if (data.disposition) {\n attachment.contentDisposition = data.disposition;\n }\n\n if (data.filename) {\n attachment.filename = data.filename;\n }\n\n if (node.headers.has('content-id')) {\n attachment.contentId = [].concat(node.headers.get('content-id') || []).shift();\n attachment.cid = attachment.contentId.trim().replace(/^<|>$/g, '').trim();\n // check if the attachment is \"related\" to text content like an embedded image etc\n let parentNode = node;\n while ((parentNode = parentNode.parent)) {\n if (parentNode.contentType === 'multipart/related') {\n attachment.related = true;\n }\n }\n }\n\n attachment.headers = node.headers;\n this.push(attachment);\n this.attachmentList.push(attachment);\n } else if (node.disposition === 'inline') {\n let chunks = [];\n let chunklen = 0;\n node.contentStream = node.decoder;\n\n if (node.contentType === 'text/plain') {\n this.hasText = true;\n } else if (node.contentType === 'text/html') {\n this.hasHtml = true;\n } else if (node.contentType === 'message/delivery-status' && !this.options.keepDeliveryStatus) {\n this.hasText = true;\n }\n\n if (node.node.flowed) {\n let contentStream = node.contentStream;\n let flowDecoder = new FlowedDecoder({\n delSp: node.node.delSp\n });\n contentStream.on('error', err => {\n flowDecoder.emit('error', err);\n });\n contentStream.pipe(flowDecoder);\n node.contentStream = flowDecoder;\n }\n\n let charset = node.charset || 'utf-8';\n //charset = charset || 'windows-1257';\n\n if (!['ascii', 'usascii', 'utf8'].includes(charset.toLowerCase().replace(/[^a-z0-9]+/g, ''))) {\n try {\n let contentStream = node.contentStream;\n let decodeStream = this.decoder.decodeStream(charset);\n contentStream.on('error', err => {\n decodeStream.emit('error', err);\n });\n contentStream.pipe(decodeStream);\n node.contentStream = decodeStream;\n } catch (E) {\n // do not decode charset\n }\n }\n\n node.contentStream.on('readable', () => {\n let chunk;\n while ((chunk = node.contentStream.read()) !== null) {\n if (typeof chunk === 'string') {\n chunk = Buffer.from(chunk);\n }\n chunks.push(chunk);\n chunklen += chunk.length;\n }\n });\n\n node.contentStream.once('end', () => {\n node.textContent = Buffer.concat(chunks, chunklen).toString().replace(/\\r?\\n/g, '\\n');\n });\n\n node.contentStream.once('error', err => {\n this.emit('error', err);\n });\n }\n\n break;\n }\n\n case 'data':\n if (this.curnode && this.curnode.decoder) {\n this.curnode.decoder.end();\n }\n\n if (this.waitUntilAttachmentEnd) {\n this.attachmentCallback = done;\n return;\n }\n\n // multipart message structure\n // this is not related to any specific 'node' block as it includes\n // everything between the end of some node body and between the next header\n //process.stdout.write(data.value);\n break;\n\n case 'body':\n if (this.curnode && this.curnode.decoder && this.curnode.decoder.writable) {\n if (this.curnode.decoder.write(data.value) === false) {\n return this.curnode.decoder.once('drain', done);\n }\n }\n\n // Leaf element body. Includes the body for the last 'node' block. You might\n // have several 'body' calls for a single 'node' block\n //process.stdout.write(data.value);\n break;\n }\n\n setImmediate(done);\n }\n\n _getPartId(parentBoundary) {\n let boundaryIndex = this.boundaries.findIndex(item => item.name === parentBoundary);\n if (boundaryIndex === -1) {\n this.boundaries.push({ name: parentBoundary, count: 1 });\n boundaryIndex = this.boundaries.length - 1;\n } else {\n this.boundaries[boundaryIndex].count++;\n }\n let partId = '1';\n for (let i = 0; i <= boundaryIndex; i++) {\n if (i === 0) partId = this.boundaries[i].count.toString();\n else partId += '.' + this.boundaries[i].count.toString();\n }\n return partId;\n }\n\n getAddressesHTML(value) {\n let formatSingleLevel = addresses =>\n addresses\n .map(address => {\n let str = '<span class=\"mp_address_group\">';\n if (address.name) {\n str += '<span class=\"mp_address_name\">' + he.encode(address.name) + (address.group ? ': ' : '') + '</span>';\n }\n if (address.address) {\n let link = '<a href=\"mailto:' + he.encode(address.address) + '\" class=\"mp_address_email\">' + he.encode(address.address) + '</a>';\n if (address.name) {\n str += ' <' + link + '>';\n } else {\n str += link;\n }\n }\n if (address.group) {\n str += formatSingleLevel(address.group) + ';';\n }\n return str + '</span>';\n })\n .join(', ');\n return formatSingleLevel([].concat(value || []));\n }\n\n getAddressesText(value) {\n let formatSingleLevel = addresses =>\n addresses\n .map(address => {\n let str = '';\n if (address.name) {\n str += `\"${address.name}\"` + (address.group ? ': ' : '');\n }\n if (address.address) {\n let link = address.address;\n if (address.name) {\n str += ' <' + link + '>';\n } else {\n str += link;\n }\n }\n if (address.group) {\n str += formatSingleLevel(address.group) + ';';\n }\n return str;\n })\n .join(', ');\n return formatSingleLevel([].concat(value || []));\n }\n\n updateImageLinks(replaceCallback, done) {\n if (!this.html) {\n return setImmediate(() => done(null, false));\n }\n\n let cids = new Map();\n let html = (this.html || '').toString();\n\n if (this.options.skipImageLinks) {\n return done(null, html);\n }\n\n html.replace(/\\bcid:([^'\"\\s]{1,256})/g, (match, cid) => {\n for (let i = 0, len = this.attachmentList.length; i < len; i++) {\n if (this.attachmentList[i].cid === cid && /^image\\/[\\w]+$/i.test(this.attachmentList[i].contentType)) {\n cids.set(cid, {\n attachment: this.attachmentList[i]\n });\n break;\n }\n }\n return match;\n });\n\n let cidList = [];\n cids.forEach(entry => {\n cidList.push(entry);\n });\n\n let pos = 0;\n let processNext = () => {\n if (pos >= cidList.length) {\n html = html.replace(/\\bcid:([^'\"\\s]{1,256})/g, (match, cid) => {\n if (cids.has(cid) && cids.get(cid).url) {\n return cids.get(cid).url;\n }\n return match;\n });\n\n return done(null, html);\n }\n let entry = cidList[pos++];\n replaceCallback(entry.attachment, (err, url) => {\n if (err) {\n return setImmediate(() => done(err));\n }\n entry.url = url;\n setImmediate(processNext);\n });\n };\n\n setImmediate(processNext);\n }\n\n textToHtml(str) {\n if (this.options.skipTextToHtml) {\n return '';\n }\n str = (str || '').toString();\n let encoded;\n\n let linkified = false;\n if (!this.options.skipTextLinks) {\n try {\n if (linkify.pretest(str)) {\n linkified = true;\n let links = linkify.match(str) || [];\n let result = [];\n let last = 0;\n\n links.forEach(link => {\n if (last < link.index) {\n let textPart = he\n // encode special chars\n .encode(str.slice(last, link.index), {\n useNamedReferences: true\n });\n result.push(textPart);\n }\n\n result.push(`<a href=\"${link.url}\">${link.text}</a>`);\n\n last = link.lastIndex;\n });\n\n let textPart = he\n // encode special chars\n .encode(str.slice(last), {\n useNamedReferences: true\n });\n result.push(textPart);\n\n encoded = result.join('');\n }\n } catch (E) {\n // failed, don't linkify\n }\n }\n\n if (!linkified) {\n encoded = he\n // encode special chars\n .encode(str, {\n useNamedReferences: true\n });\n }\n\n let text =\n '<p>' +\n encoded\n .replace(/\\r?\\n/g, '\\n')\n .trim() // normalize line endings\n .replace(/[ \\t]+$/gm, '')\n .trim() // trim empty line endings\n .replace(/\\n\\n+/g, '</p><p>')\n .trim() // insert <p> to multiple linebreaks\n .replace(/\\n/g, '<br/>') + // insert <br> to single linebreaks\n '</p>';\n\n return text;\n }\n}\n\nmodule.exports = MailParser;\n//# sourceURL=[module]\n//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"file":"(action-browser)/./node_modules/mailparser/lib/mail-parser.js","mappings":"AAAa;;AAEb,kBAAkB,mBAAO,CAAC,qEAAW;AACrC,gBAAgB,mBAAO,CAAC,uEAAS;AACjC,sBAAsB,mBAAO,CAAC,mIAA8B;AAC5D,kBAAkB,uDAA2B;AAC7C;AACA,iBAAiB,mBAAO,CAAC,gFAAa;AACtC,sBAAsB,mBAAO,CAAC,qGAA8B;AAC5D,mBAAmB,mBAAO,CAAC,oFAAe;AAC1C,cAAc,mBAAO,CAAC,2EAAY;AAClC,QAAQ,aAAa,EAAE,mBAAO,CAAC,uFAAc;AAC7C,WAAW,mBAAO,CAAC,oDAAI;AACvB,gBAAgB,mBAAO,CAAC,iFAAY;AACpC,aAAa,mBAAO,CAAC,6DAAM;AAC3B,yBAAyB,mBAAO,CAAC,yFAAmB;;AAEpD;AACA;AACA;AACA;AACA;AACA,WAAW,kDAAkD;;AAE7D;AACA;AACA;AACA;AACA;;AAEA;AACA,0DAA0D,KAAK;AAC/D;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,KAAK;AACL;AACA;AACA;AACA,CAAC;;AAED;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,SAAS;AACT;;AAEA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;;AAEA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,aAAa;AACb;AACA;AACA;AACA;AACA,UAAU;AACV;AACA;AACA;;AAEA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA,SAAS;;AAET;AACA;AACA;AACA;;AAEA;AACA;;AAEA;AACA;AACA;;AAEA;AACA;AACA;AACA;;AAEA;AACA;;AAEA;AACA;AACA;;AAEA;;AAEA;;AAEA;;AAEA;;AAEA;AACA;AACA;AACA;AACA;AACA,SAAS;;AAET;AACA;AACA;AACA;AACA;AACA,SAAS;;AAET;AACA;AACA;AACA;AACA;AACA;AACA,SAAS;;AAET,6CAA6C,2BAA2B;AACxE;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,UAAU;AACV;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,SAAS;AACT;;AAEA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA,aAAa;AACb,UAAU;AACV;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,aAAa;AACb;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA,cAAc;AACd;AACA;;AAEA;AACA;;AAEA;AACA;AACA;AACA,aAAa;AACb;AACA,UAAU;AACV;AACA;AACA,aAAa;AACb;AACA;;AAEA;AACA;AACA;AACA;AACA,oEAAoE;AACpE;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,6DAA6D;AAC7D;AACA;AACA,0BAA0B;AAC1B;AACA;AACA,qBAAqB;AACrB;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,sBAAsB;AACtB;AACA;AACA;AACA;AACA;AACA;AACA,sBAAsB;AACtB;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,sBAAsB;AACtB;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA,kBAAkB;AAClB;AACA,kBAAkB;AAClB;AACA;AACA;AACA,SAAS;;AAET;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA,kBAAkB;AAClB;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA,yCAAyC;AACzC;AACA,qBAAqB;AACrB,iBAAiB;AACjB;AACA;AACA,SAAS;;AAET;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,kBAAkB;AAClB;AACA;AACA;AACA;AACA,kBAAkB;AAClB;AACA,kBAAkB;AAClB;AACA;AACA;AACA;AACA;AACA;AACA,aAAa;AACb;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,cAAc;AACd;AACA,cAAc;AACd;AACA;AACA,UAAU;AACV;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;;AAEA;AACA;AACA;;AAEA;AACA;AACA;;AAEA;AACA;;AAEA;AACA;AACA,wBAAwB,sBAAsB;AAC9C;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA,qBAAqB;AACrB;;AAEA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA,kBAAkB;AAClB;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,kBAAkB;AAClB,oDAAoD;AACpD;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;;AAEA;AACA;AACA,cAAc;AACd;AACA;;AAEA;;AAEA;;AAEA;AACA;AACA;;AAEA;AACA;AACA;AACA,aAAa;AACb;AACA;;AAEA;AACA;AACA;AACA;;AAEA;;AAEA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,qBAAqB;AACrB;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA,iCAAiC;AACjC;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,iCAAiC;AACjC;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,kBAAkB;AAClB;AACA;AACA;AACA;AACA,kBAAkB;AAClB;AACA;AACA;AACA;AACA,0BAA0B;AAC1B,sFAAsF,yBAAyB;AAC/G;AACA;AACA,0BAA0B;AAC1B;AACA;AACA,8BAA8B;AAC9B;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,iBAAiB;AACjB;AACA;;AAEA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,qBAAqB;AACrB;;AAEA;AACA;AACA;AACA;;AAEA;AACA;AACA;;AAEA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA,qBAAqB;;AAErB;AACA;;AAEA;AACA;AACA;AACA,qBAAqB;;AAErB;AACA;AACA,qBAAqB;;AAErB;AACA;;AAEA;AACA;AACA;AACA;;AAEA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA,kBAAkB;AAClB;AACA;AACA;;AAEA;AACA;AACA,sBAAsB;AACtB;AACA,sBAAsB;AACtB;AACA;;AAEA;AACA;AACA;AACA;AACA,yBAAyB;AACzB;AACA;AACA,yBAAyB;AACzB;AACA;AACA;;AAEA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA,6BAA6B;AAC7B;AACA;AACA,0BAA0B;AAC1B;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,qBAAqB;;AAErB;AACA;AACA,qBAAqB;;AAErB;AACA;AACA,qBAAqB;AACrB;;AAEA;AACA;;AAEA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;;AAEA;AACA;;AAEA;AACA;AACA;AACA,mCAAmC,gCAAgC;AACnE;AACA,UAAU;AACV;AACA;AACA;AACA,wBAAwB,oBAAoB;AAC5C;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,yCAAyC,gBAAgB;AACzD,0BAA0B;AAC1B;AACA;AACA;AACA;AACA,oEAAoE;AACpE;AACA;AACA,iBAAiB;AACjB;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA,mCAAmC,aAAa;AAChD;AACA;AACA;AACA;AACA;AACA,0BAA0B;AAC1B;AACA;AACA;AACA;AACA,oEAAoE;AACpE;AACA;AACA,iBAAiB;AACjB;AACA;AACA;;AAEA;AACA;AACA;AACA;;AAEA;AACA;;AAEA;AACA;AACA;;AAEA,qCAAqC,MAAM;AAC3C,8DAA8D,SAAS;AACvE;AACA;AACA;AACA,qBAAqB;AACrB;AACA;AACA;AACA;AACA,SAAS;;AAET;AACA;AACA;AACA,SAAS;;AAET;AACA;AACA;AACA,oDAAoD,MAAM;AAC1D;AACA;AACA;AACA;AACA,iBAAiB;;AAEjB;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,aAAa;AACb;;AAEA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA,iCAAiC;AACjC;AACA;;AAEA,gDAAgD,SAAS,IAAI,UAAU;;AAEvE;AACA,qBAAqB;;AAErB;AACA;AACA;AACA;AACA,yBAAyB;AACzB;;AAEA;AACA;AACA,cAAc;AACd;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA,iBAAiB;AACjB;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;;AAEA","sources":["/home/alma/nextgen/Neah-mail/node_modules/mailparser/lib/mail-parser.js"],"sourcesContent":["'use strict';\n\nconst mailsplit = require('mailsplit');\nconst libmime = require('libmime');\nconst addressparser = require('nodemailer/lib/addressparser');\nconst Transform = require('stream').Transform;\nconst Splitter = mailsplit.Splitter;\nconst punycode = require('punycode.js');\nconst FlowedDecoder = require('mailsplit/lib/flowed-decoder');\nconst StreamHash = require('./stream-hash');\nconst iconv = require('iconv-lite');\nconst { htmlToText } = require('html-to-text');\nconst he = require('he');\nconst linkify = require('linkify-it')();\nconst tlds = require('tlds');\nconst encodingJapanese = require('encoding-japanese');\n\nlinkify\n    .tlds(tlds) // Reload with full tlds list\n    .tlds('onion', true) // Add unofficial `.onion` domain\n    .add('git:', 'http:') // Add `git:` ptotocol as \"alias\"\n    .add('ftp:', null) // Disable `ftp:` ptotocol\n    .set({ fuzzyIP: true, fuzzyLink: true, fuzzyEmail: true });\n\n// twitter linkifier from\n// https://github.com/markdown-it/linkify-it#example-2-add-twitter-mentions-handler\nlinkify.add('@', {\n    validate(text, pos, self) {\n        let tail = text.slice(pos);\n\n        if (!self.re.twitter) {\n            self.re.twitter = new RegExp('^([a-zA-Z0-9_]){1,15}(?!_)(?=$|' + self.re.src_ZPCc + ')');\n        }\n        if (self.re.twitter.test(tail)) {\n            // Linkifier allows punctuation chars before prefix,\n            // but we additionally disable `@` (\"@@mention\" is invalid)\n            if (pos >= 2 && tail[pos - 2] === '@') {\n                return false;\n            }\n            return tail.match(self.re.twitter)[0].length;\n        }\n        return 0;\n    },\n    normalize(match) {\n        match.url = 'https://twitter.com/' + match.url.replace(/^@/, '');\n    }\n});\n\nclass IconvDecoder extends Transform {\n    constructor(Iconv, charset) {\n        super();\n\n        // Iconv throws error on ks_c_5601-1987 when it is mapped to EUC-KR\n        // https://github.com/bnoordhuis/node-iconv/issues/169\n        if (charset.toLowerCase() === 'ks_c_5601-1987') {\n            charset = 'CP949';\n        }\n        this.stream = new Iconv(charset, 'UTF-8//TRANSLIT//IGNORE');\n\n        this.inputEnded = false;\n        this.endCb = false;\n\n        this.stream.on('error', err => this.emit('error', err));\n        this.stream.on('data', chunk => this.push(chunk));\n        this.stream.on('end', () => {\n            this.inputEnded = true;\n            if (typeof this.endCb === 'function') {\n                this.endCb();\n            }\n        });\n    }\n\n    _transform(chunk, encoding, done) {\n        this.stream.write(chunk);\n        done();\n    }\n\n    _flush(done) {\n        this.endCb = done;\n        this.stream.end();\n    }\n}\n\nclass JPDecoder extends Transform {\n    constructor(charset) {\n        super();\n\n        this.charset = charset;\n        this.chunks = [];\n        this.chunklen = 0;\n    }\n\n    _transform(chunk, encoding, done) {\n        if (typeof chunk === 'string') {\n            chunk = Buffer.from(chunk, encoding);\n        }\n\n        this.chunks.push(chunk);\n        this.chunklen += chunk.length;\n        done();\n    }\n\n    _flush(done) {\n        let input = Buffer.concat(this.chunks, this.chunklen);\n        try {\n            let output = encodingJapanese.convert(input, {\n                to: 'UNICODE', // to_encoding\n                from: this.charset, // from_encoding\n                type: 'string'\n            });\n            if (typeof output === 'string') {\n                output = Buffer.from(output);\n            }\n            this.push(output);\n        } catch (err) {\n            // keep as is on errors\n            this.push(input);\n        }\n\n        done();\n    }\n}\n\nclass MailParser extends Transform {\n    constructor(config) {\n        super({\n            readableObjectMode: true,\n            writableObjectMode: false\n        });\n\n        this.options = config || {};\n        this.splitter = new Splitter(config);\n        this.finished = false;\n        this.waitingEnd = false;\n\n        this.headers = false;\n        this.headerLines = false;\n\n        this.endReceived = false;\n        this.reading = false;\n        this.hasFailed = false;\n\n        this.tree = false;\n        this.curnode = false;\n        this.waitUntilAttachmentEnd = false;\n        this.attachmentCallback = false;\n\n        this.hasHtml = false;\n        this.hasText = false;\n\n        this.text = false;\n        this.html = false;\n        this.textAsHtml = false;\n\n        this.attachmentList = [];\n\n        this.boundaries = [];\n\n        this.textTypes = ['text/plain', 'text/html'].concat(!this.options.keepDeliveryStatus ? 'message/delivery-status' : []);\n\n        this.decoder = this.getDecoder();\n\n        this.splitter.on('readable', () => {\n            if (this.reading) {\n                return false;\n            }\n            this.readData();\n        });\n\n        this.splitter.on('end', () => {\n            this.endReceived = true;\n            if (!this.reading) {\n                this.endStream();\n            }\n        });\n\n        this.splitter.on('error', err => {\n            this.hasFailed = true;\n            if (typeof this.waitingEnd === 'function') {\n                return this.waitingEnd(err);\n            }\n            this.emit('error', err);\n        });\n\n        this.libmime = new libmime.Libmime({ Iconv: this.options.Iconv });\n    }\n\n    getDecoder() {\n        if (this.options.Iconv) {\n            const Iconv = this.options.Iconv;\n            // create wrapper\n            return {\n                decodeStream(charset) {\n                    return new IconvDecoder(Iconv, charset);\n                }\n            };\n        } else {\n            return {\n                decodeStream(charset) {\n                    charset = (charset || 'ascii').toString().trim().toLowerCase();\n                    if (/^jis|^iso-?2022-?jp|^EUCJP/i.test(charset)) {\n                        // special case not supported by iconv-lite\n                        return new JPDecoder(charset);\n                    }\n\n                    return iconv.decodeStream(charset);\n                }\n            };\n        }\n    }\n\n    readData() {\n        if (this.hasFailed) {\n            return false;\n        }\n        this.reading = true;\n        let data = this.splitter.read();\n        if (data === null) {\n            this.reading = false;\n            if (this.endReceived) {\n                this.endStream();\n            }\n            return;\n        }\n\n        this.processChunk(data, err => {\n            if (err) {\n                if (typeof this.waitingEnd === 'function') {\n                    return this.waitingEnd(err);\n                }\n                return this.emit('error', err);\n            }\n            setImmediate(() => this.readData());\n        });\n    }\n\n    endStream() {\n        this.finished = true;\n\n        if (this.curnode && this.curnode.decoder) {\n            this.curnode.decoder.end();\n        }\n        if (typeof this.waitingEnd === 'function') {\n            this.waitingEnd();\n        }\n    }\n\n    _transform(chunk, encoding, done) {\n        if (!chunk || !chunk.length) {\n            return done();\n        }\n\n        if (this.splitter.write(chunk) === false) {\n            return this.splitter.once('drain', () => {\n                done();\n            });\n        } else {\n            return done();\n        }\n    }\n\n    _flush(done) {\n        setImmediate(() => this.splitter.end());\n        if (this.finished) {\n            return this.cleanup(done);\n        }\n        this.waitingEnd = () => {\n            this.cleanup(() => {\n                done();\n            });\n        };\n    }\n\n    cleanup(done) {\n        let finish = () => {\n            try {\n                let t = this.getTextContent();\n                this.push(t);\n            } catch (err) {\n                return this.emit('error', err);\n            }\n\n            done();\n        };\n\n        if (this.curnode && this.curnode.decoder && this.curnode.decoder.readable && !this.decoderEnded) {\n            (this.curnode.contentStream || this.curnode.decoder).once('end', () => {\n                finish();\n            });\n            this.curnode.decoder.end();\n        } else {\n            setImmediate(() => {\n                finish();\n            });\n        }\n    }\n\n    processHeaders(lines) {\n        let headers = new Map();\n        (lines || []).forEach(line => {\n            let key = line.key;\n            let value = ((this.libmime.decodeHeader(line.line) || {}).value || '').toString().trim();\n            value = Buffer.from(value, 'binary').toString();\n            switch (key) {\n                case 'content-type':\n                case 'content-disposition':\n                case 'dkim-signature':\n                    value = this.libmime.parseHeaderValue(value);\n                    if (value.value) {\n                        value.value = this.libmime.decodeWords(value.value);\n                    }\n                    Object.keys((value && value.params) || {}).forEach(key => {\n                        try {\n                            value.params[key] = this.libmime.decodeWords(value.params[key]);\n                        } catch (E) {\n                            // ignore, keep as is\n                        }\n                    });\n                    break;\n                case 'date': {\n                    let dateValue = new Date(value);\n                    if (isNaN(dateValue)) {\n                        // date parsing failed :S\n                        dateValue = new Date();\n                    }\n                    value = dateValue;\n                    break;\n                }\n                case 'subject':\n                    try {\n                        value = this.libmime.decodeWords(value);\n                    } catch (E) {\n                        // ignore, keep as is\n                    }\n                    break;\n                case 'references':\n                    try {\n                        value = this.libmime.decodeWords(value);\n                    } catch (E) {\n                        // ignore\n                    }\n                    value = value.split(/\\s+/).map(this.ensureMessageIDFormat);\n                    break;\n                case 'message-id':\n                case 'in-reply-to':\n                    try {\n                        value = this.libmime.decodeWords(value);\n                    } catch (E) {\n                        // ignore\n                    }\n                    value = this.ensureMessageIDFormat(value);\n                    break;\n                case 'priority':\n                case 'x-priority':\n                case 'x-msmail-priority':\n                case 'importance':\n                    key = 'priority';\n                    value = this.parsePriority(value);\n                    break;\n                case 'from':\n                case 'to':\n                case 'cc':\n                case 'bcc':\n                case 'sender':\n                case 'reply-to':\n                case 'delivered-to':\n                case 'return-path':\n                case 'disposition-notification-to':\n                    value = addressparser(value);\n                    this.decodeAddresses(value);\n                    value = {\n                        value,\n                        html: this.getAddressesHTML(value),\n                        text: this.getAddressesText(value)\n                    };\n                    break;\n            }\n\n            // handle list-* keys\n            if (key.substr(0, 5) === 'list-') {\n                value = this.parseListHeader(key.substr(5), value);\n                key = 'list';\n            }\n\n            if (value) {\n                if (!headers.has(key)) {\n                    headers.set(key, [].concat(value || []));\n                } else if (Array.isArray(value)) {\n                    headers.set(key, headers.get(key).concat(value));\n                } else {\n                    headers.get(key).push(value);\n                }\n            }\n        });\n\n        // keep only the first value\n        let singleKeys = [\n            'message-id',\n            'content-id',\n            'from',\n            'sender',\n            'in-reply-to',\n            'reply-to',\n            'subject',\n            'date',\n            'content-disposition',\n            'content-type',\n            'content-transfer-encoding',\n            'priority',\n            'mime-version',\n            'content-description',\n            'precedence',\n            'errors-to',\n            'disposition-notification-to'\n        ];\n\n        headers.forEach((value, key) => {\n            if (Array.isArray(value)) {\n                if (singleKeys.includes(key) && value.length) {\n                    headers.set(key, value[value.length - 1]);\n                } else if (value.length === 1) {\n                    headers.set(key, value[0]);\n                }\n            }\n\n            if (key === 'list') {\n                // normalize List-* headers\n                let listValue = {};\n                [].concat(value || []).forEach(val => {\n                    Object.keys(val || {}).forEach(listKey => {\n                        listValue[listKey] = val[listKey];\n                    });\n                });\n                headers.set(key, listValue);\n            }\n        });\n\n        return headers;\n    }\n\n    parseListHeader(key, value) {\n        let addresses = addressparser(value);\n        let response = {};\n        let data = addresses\n            .map(address => {\n                if (/^https?:/i.test(address.name)) {\n                    response.url = address.name;\n                } else if (address.name) {\n                    response.name = address.name;\n                }\n                if (/^mailto:/.test(address.address)) {\n                    response.mail = address.address.substr(7);\n                } else if (address.address && address.address.indexOf('@') < 0) {\n                    response.id = address.address;\n                } else if (address.address) {\n                    response.mail = address.address;\n                }\n                if (Object.keys(response).length) {\n                    return response;\n                }\n                return false;\n            })\n            .filter(address => address);\n        if (data.length) {\n            return {\n                [key]: response\n            };\n        }\n        return false;\n    }\n\n    parsePriority(value) {\n        value = value.toLowerCase().trim();\n        if (!isNaN(parseInt(value, 10))) {\n            // support \"X-Priority: 1 (Highest)\"\n            value = parseInt(value, 10) || 0;\n            if (value === 3) {\n                return 'normal';\n            } else if (value > 3) {\n                return 'low';\n            } else {\n                return 'high';\n            }\n        } else {\n            switch (value) {\n                case 'non-urgent':\n                case 'low':\n                    return 'low';\n                case 'urgent':\n                case 'high':\n                    return 'high';\n            }\n        }\n        return 'normal';\n    }\n\n    ensureMessageIDFormat(value) {\n        if (!value.length) {\n            return false;\n        }\n\n        if (value.charAt(0) !== '<') {\n            value = '<' + value;\n        }\n\n        if (value.charAt(value.length - 1) !== '>') {\n            value += '>';\n        }\n\n        return value;\n    }\n\n    decodeAddresses(addresses) {\n        let processedAddress = new WeakSet();\n        for (let i = 0; i < addresses.length; i++) {\n            let address = addresses[i];\n            address.name = (address.name || '').toString().trim();\n\n            if (!address.address && /^(=\\?([^?]+)\\?[Bb]\\?[^?]*\\?=)(\\s*=\\?([^?]+)\\?[Bb]\\?[^?]*\\?=)*$/.test(address.name) && !processedAddress.has(address)) {\n                let parsed = addressparser(this.libmime.decodeWords(address.name));\n                if (parsed.length) {\n                    parsed.forEach(entry => {\n                        processedAddress.add(entry);\n                        addresses.push(entry);\n                    });\n                }\n\n                // remove current element\n                addresses.splice(i, 1);\n                i--;\n                continue;\n            }\n\n            if (address.name) {\n                try {\n                    address.name = this.libmime.decodeWords(address.name);\n                } catch (E) {\n                    //ignore, keep as is\n                }\n            }\n            if (/@xn--/.test(address.address)) {\n                try {\n                    address.address =\n                        address.address.substr(0, address.address.lastIndexOf('@') + 1) +\n                        punycode.toUnicode(address.address.substr(address.address.lastIndexOf('@') + 1));\n                } catch (E) {\n                    // Not a valid punycode string; keep as is\n                }\n            }\n            if (address.group) {\n                this.decodeAddresses(address.group);\n            }\n        }\n    }\n\n    createNode(node) {\n        let contentType = node.contentType;\n        let disposition = node.disposition;\n        let encoding = node.encoding;\n        let charset = node.charset;\n\n        if (!contentType && node.root) {\n            contentType = 'text/plain';\n        }\n\n        let newNode = {\n            node,\n            headerLines: node.headers.lines,\n            headers: this.processHeaders(node.headers.getList()),\n            contentType,\n            children: []\n        };\n\n        if (!/^multipart\\//i.test(contentType)) {\n            if (disposition && !['attachment', 'inline'].includes(disposition)) {\n                disposition = 'attachment';\n            }\n\n            if (!disposition && !this.textTypes.includes(contentType)) {\n                newNode.disposition = 'attachment';\n            } else {\n                newNode.disposition = disposition || 'inline';\n            }\n\n            newNode.isAttachment = !this.textTypes.includes(contentType) || newNode.disposition !== 'inline';\n\n            newNode.encoding = ['quoted-printable', 'base64'].includes(encoding) ? encoding : 'binary';\n\n            if (charset) {\n                newNode.charset = charset;\n            }\n\n            let decoder = node.getDecoder();\n            decoder.on('end', () => {\n                this.decoderEnded = true;\n            });\n            newNode.decoder = decoder;\n        }\n\n        if (node.root) {\n            this.headers = newNode.headers;\n            this.headerLines = newNode.headerLines;\n        }\n\n        // find location in tree\n\n        if (!this.tree) {\n            newNode.root = true;\n            this.curnode = this.tree = newNode;\n            return newNode;\n        }\n\n        // immediate child of root node\n        if (!this.curnode.parent) {\n            newNode.parent = this.curnode;\n            this.curnode.children.push(newNode);\n            this.curnode = newNode;\n            return newNode;\n        }\n\n        // siblings\n        if (this.curnode.parent.node === node.parentNode) {\n            newNode.parent = this.curnode.parent;\n            this.curnode.parent.children.push(newNode);\n            this.curnode = newNode;\n            return newNode;\n        }\n\n        // first child\n        if (this.curnode.node === node.parentNode) {\n            newNode.parent = this.curnode;\n            this.curnode.children.push(newNode);\n            this.curnode = newNode;\n            return newNode;\n        }\n\n        // move up\n        let parentNode = this.curnode;\n        while ((parentNode = parentNode.parent)) {\n            if (parentNode.node === node.parentNode) {\n                newNode.parent = parentNode;\n                parentNode.children.push(newNode);\n                this.curnode = newNode;\n                return newNode;\n            }\n        }\n\n        // should never happen, can't detect parent\n        this.curnode = newNode;\n        return newNode;\n    }\n\n    getTextContent() {\n        let text = [];\n        let html = [];\n        let processNode = (alternative, level, node) => {\n            if (node.showMeta) {\n                let meta = ['From', 'Subject', 'Date', 'To', 'Cc', 'Bcc']\n                    .map(fkey => {\n                        let key = fkey.toLowerCase();\n                        if (!node.headers.has(key)) {\n                            return false;\n                        }\n                        let value = node.headers.get(key);\n                        if (!value) {\n                            return false;\n                        }\n                        return {\n                            key: fkey,\n                            value: Array.isArray(value) ? value[value.length - 1] : value\n                        };\n                    })\n                    .filter(entry => entry);\n                if (this.hasHtml) {\n                    html.push(\n                        '<table class=\"mp_head\">' +\n                            meta\n                                .map(entry => {\n                                    let value = entry.value;\n                                    switch (entry.key) {\n                                        case 'From':\n                                        case 'To':\n                                        case 'Cc':\n                                        case 'Bcc':\n                                            value = value.html;\n                                            break;\n                                        case 'Date':\n                                            value = this.options.formatDateString ? this.options.formatDateString(value) : value.toUTCString();\n                                            break;\n                                        case 'Subject':\n                                            value = '<strong>' + he.encode(value) + '</strong>';\n                                            break;\n                                        default:\n                                            value = he.encode(value);\n                                    }\n\n                                    return '<tr><td class=\"mp_head_key\">' + he.encode(entry.key) + ':</td><td class=\"mp_head_value\">' + value + '<td></tr>';\n                                })\n                                .join('\\n') +\n                            '<table>'\n                    );\n                }\n                if (this.hasText) {\n                    text.push(\n                        '\\n' +\n                            meta\n                                .map(entry => {\n                                    let value = entry.value;\n                                    switch (entry.key) {\n                                        case 'From':\n                                        case 'To':\n                                        case 'Cc':\n                                        case 'Bcc':\n                                            value = value.text;\n                                            break;\n                                        case 'Date':\n                                            value = this.options.formatDateString ? this.options.formatDateString(value) : value.toUTCString();\n                                            break;\n                                    }\n                                    return entry.key + ': ' + value;\n                                })\n                                .join('\\n') +\n                            '\\n'\n                    );\n                }\n            }\n            if (node.textContent) {\n                if (node.contentType === 'text/plain') {\n                    text.push(node.textContent);\n                    if (!alternative && this.hasHtml) {\n                        html.push(this.textToHtml(node.textContent));\n                    }\n                } else if (node.contentType === 'message/delivery-status' && !this.options.keepDeliveryStatus) {\n                    text.push(node.textContent);\n                    if (!alternative && this.hasHtml) {\n                        html.push(this.textToHtml(node.textContent));\n                    }\n                } else if (node.contentType === 'text/html') {\n                    let failedToParseHtml = false;\n                    if ((!alternative && this.hasText) || (node.root && !this.hasText)) {\n                        if (this.options.skipHtmlToText) {\n                            text.push('');\n                        } else if (node.textContent.length > this.options.maxHtmlLengthToParse) {\n                            this.emit('error', new Error(`HTML too long for parsing ${node.textContent.length} bytes`));\n                            text.push('Invalid HTML content (too long)');\n                            failedToParseHtml = true;\n                        } else {\n                            try {\n                                text.push(htmlToText(node.textContent));\n                            } catch (err) {\n                                this.emit('error', new Error('Failed to parse HTML'));\n                                text.push('Invalid HTML content');\n                                failedToParseHtml = true;\n                            }\n                        }\n                    }\n                    if (!failedToParseHtml) {\n                        html.push(node.textContent);\n                    }\n                }\n            }\n            alternative = alternative || node.contentType === 'multipart/alternative';\n            if (node.children) {\n                node.children.forEach(subNode => {\n                    processNode(alternative, level + 1, subNode);\n                });\n            }\n        };\n\n        processNode(false, 0, this.tree);\n\n        let response = {\n            type: 'text'\n        };\n        if (html.length) {\n            this.html = response.html = html.join('<br/>\\n');\n        }\n        if (text.length) {\n            this.text = response.text = text.join('\\n');\n            this.textAsHtml = response.textAsHtml = text.map(part => this.textToHtml(part)).join('<br/>\\n');\n        }\n        return response;\n    }\n\n    processChunk(data, done) {\n        let partId = null;\n        if (data._parentBoundary) {\n            partId = this._getPartId(data._parentBoundary);\n        }\n        switch (data.type) {\n            case 'node': {\n                let node = this.createNode(data);\n                if (node === this.tree) {\n                    ['subject', 'references', 'date', 'to', 'from', 'to', 'cc', 'bcc', 'message-id', 'in-reply-to', 'reply-to'].forEach(key => {\n                        if (node.headers.has(key)) {\n                            this[key.replace(/-([a-z])/g, (m, c) => c.toUpperCase())] = node.headers.get(key);\n                        }\n                    });\n                    this.emit('headers', node.headers);\n\n                    if (node.headerLines) {\n                        this.emit('headerLines', node.headerLines);\n                    }                    \n                }\n\n                if (data.contentType === 'message/rfc822' && data.messageNode) {\n                    break;\n                }\n\n                if (data.parentNode && data.parentNode.contentType === 'message/rfc822') {\n                    node.showMeta = true;\n                }\n\n                if (node.isAttachment) {\n                    let contentType = node.contentType;\n                    if (node.contentType === 'application/octet-stream' && data.filename) {\n                        contentType = this.libmime.detectMimeType(data.filename) || 'application/octet-stream';\n                    }\n\n                    let attachment = {\n                        type: 'attachment',\n                        content: null,\n                        contentType,\n                        partId,\n                        release: () => {\n                            attachment.release = null;\n                            if (this.waitUntilAttachmentEnd && typeof this.attachmentCallback === 'function') {\n                                setImmediate(this.attachmentCallback);\n                            }\n                            this.attachmentCallback = false;\n                            this.waitUntilAttachmentEnd = false;\n                        }\n                    };\n\n                    let algo = this.options.checksumAlgo || 'md5';\n                    let hasher = new StreamHash(attachment, algo);\n                    node.decoder.on('error', err => {\n                        hasher.emit('error', err);\n                    });\n\n                    node.decoder.on('readable', () => {\n                        let chunk;\n\n                        while ((chunk = node.decoder.read()) !== null) {\n                            hasher.write(chunk);\n                        }\n                    });\n\n                    node.decoder.once('end', () => {\n                        hasher.end();\n                    });\n\n                    //node.decoder.pipe(hasher);\n                    attachment.content = hasher;\n\n                    this.waitUntilAttachmentEnd = true;\n                    if (data.disposition) {\n                        attachment.contentDisposition = data.disposition;\n                    }\n\n                    if (data.filename) {\n                        attachment.filename = data.filename;\n                    }\n\n                    if (node.headers.has('content-id')) {\n                        attachment.contentId = [].concat(node.headers.get('content-id') || []).shift();\n                        attachment.cid = attachment.contentId.trim().replace(/^<|>$/g, '').trim();\n                        // check if the attachment is \"related\" to text content like an embedded image etc\n                        let parentNode = node;\n                        while ((parentNode = parentNode.parent)) {\n                            if (parentNode.contentType === 'multipart/related') {\n                                attachment.related = true;\n                            }\n                        }\n                    }\n\n                    attachment.headers = node.headers;\n                    this.push(attachment);\n                    this.attachmentList.push(attachment);\n                } else if (node.disposition === 'inline') {\n                    let chunks = [];\n                    let chunklen = 0;\n                    node.contentStream = node.decoder;\n\n                    if (node.contentType === 'text/plain') {\n                        this.hasText = true;\n                    } else if (node.contentType === 'text/html') {\n                        this.hasHtml = true;\n                    } else if (node.contentType === 'message/delivery-status' && !this.options.keepDeliveryStatus) {\n                        this.hasText = true;\n                    }\n\n                    if (node.node.flowed) {\n                        let contentStream = node.contentStream;\n                        let flowDecoder = new FlowedDecoder({\n                            delSp: node.node.delSp\n                        });\n                        contentStream.on('error', err => {\n                            flowDecoder.emit('error', err);\n                        });\n                        contentStream.pipe(flowDecoder);\n                        node.contentStream = flowDecoder;\n                    }\n\n                    let charset = node.charset || 'utf-8';\n                    //charset = charset || 'windows-1257';\n\n                    if (!['ascii', 'usascii', 'utf8'].includes(charset.toLowerCase().replace(/[^a-z0-9]+/g, ''))) {\n                        try {\n                            let contentStream = node.contentStream;\n                            let decodeStream = this.decoder.decodeStream(charset);\n                            contentStream.on('error', err => {\n                                decodeStream.emit('error', err);\n                            });\n                            contentStream.pipe(decodeStream);\n                            node.contentStream = decodeStream;\n                        } catch (E) {\n                            // do not decode charset\n                        }\n                    }\n\n                    node.contentStream.on('readable', () => {\n                        let chunk;\n                        while ((chunk = node.contentStream.read()) !== null) {\n                            if (typeof chunk === 'string') {\n                                chunk = Buffer.from(chunk);\n                            }\n                            chunks.push(chunk);\n                            chunklen += chunk.length;\n                        }\n                    });\n\n                    node.contentStream.once('end', () => {\n                        node.textContent = Buffer.concat(chunks, chunklen).toString().replace(/\\r?\\n/g, '\\n');\n                    });\n\n                    node.contentStream.once('error', err => {\n                        this.emit('error', err);\n                    });\n                }\n\n                break;\n            }\n\n            case 'data':\n                if (this.curnode && this.curnode.decoder) {\n                    this.curnode.decoder.end();\n                }\n\n                if (this.waitUntilAttachmentEnd) {\n                    this.attachmentCallback = done;\n                    return;\n                }\n\n                // multipart message structure\n                // this is not related to any specific 'node' block as it includes\n                // everything between the end of some node body and between the next header\n                //process.stdout.write(data.value);\n                break;\n\n            case 'body':\n                if (this.curnode && this.curnode.decoder && this.curnode.decoder.writable) {\n                    if (this.curnode.decoder.write(data.value) === false) {\n                        return this.curnode.decoder.once('drain', done);\n                    }\n                }\n\n                // Leaf element body. Includes the body for the last 'node' block. You might\n                // have several 'body' calls for a single 'node' block\n                //process.stdout.write(data.value);\n                break;\n        }\n\n        setImmediate(done);\n    }\n\n    _getPartId(parentBoundary) {\n        let boundaryIndex = this.boundaries.findIndex(item => item.name === parentBoundary);\n        if (boundaryIndex === -1) {\n            this.boundaries.push({ name: parentBoundary, count: 1 });\n            boundaryIndex = this.boundaries.length - 1;\n        } else {\n            this.boundaries[boundaryIndex].count++;\n        }\n        let partId = '1';\n        for (let i = 0; i <= boundaryIndex; i++) {\n            if (i === 0) partId = this.boundaries[i].count.toString();\n            else partId += '.' + this.boundaries[i].count.toString();\n        }\n        return partId;\n    }\n\n    getAddressesHTML(value) {\n        let formatSingleLevel = addresses =>\n            addresses\n                .map(address => {\n                    let str = '<span class=\"mp_address_group\">';\n                    if (address.name) {\n                        str += '<span class=\"mp_address_name\">' + he.encode(address.name) + (address.group ? ': ' : '') + '</span>';\n                    }\n                    if (address.address) {\n                        let link = '<a href=\"mailto:' + he.encode(address.address) + '\" class=\"mp_address_email\">' + he.encode(address.address) + '</a>';\n                        if (address.name) {\n                            str += ' &lt;' + link + '&gt;';\n                        } else {\n                            str += link;\n                        }\n                    }\n                    if (address.group) {\n                        str += formatSingleLevel(address.group) + ';';\n                    }\n                    return str + '</span>';\n                })\n                .join(', ');\n        return formatSingleLevel([].concat(value || []));\n    }\n\n    getAddressesText(value) {\n        let formatSingleLevel = addresses =>\n            addresses\n                .map(address => {\n                    let str = '';\n                    if (address.name) {\n                        str += `\"${address.name}\"` + (address.group ? ': ' : '');\n                    }\n                    if (address.address) {\n                        let link = address.address;\n                        if (address.name) {\n                            str += ' <' + link + '>';\n                        } else {\n                            str += link;\n                        }\n                    }\n                    if (address.group) {\n                        str += formatSingleLevel(address.group) + ';';\n                    }\n                    return str;\n                })\n                .join(', ');\n        return formatSingleLevel([].concat(value || []));\n    }\n\n    updateImageLinks(replaceCallback, done) {\n        if (!this.html) {\n            return setImmediate(() => done(null, false));\n        }\n\n        let cids = new Map();\n        let html = (this.html || '').toString();\n\n        if (this.options.skipImageLinks) {\n            return done(null, html);\n        }\n\n        html.replace(/\\bcid:([^'\"\\s]{1,256})/g, (match, cid) => {\n            for (let i = 0, len = this.attachmentList.length; i < len; i++) {\n                if (this.attachmentList[i].cid === cid && /^image\\/[\\w]+$/i.test(this.attachmentList[i].contentType)) {\n                    cids.set(cid, {\n                        attachment: this.attachmentList[i]\n                    });\n                    break;\n                }\n            }\n            return match;\n        });\n\n        let cidList = [];\n        cids.forEach(entry => {\n            cidList.push(entry);\n        });\n\n        let pos = 0;\n        let processNext = () => {\n            if (pos >= cidList.length) {\n                html = html.replace(/\\bcid:([^'\"\\s]{1,256})/g, (match, cid) => {\n                    if (cids.has(cid) && cids.get(cid).url) {\n                        return cids.get(cid).url;\n                    }\n                    return match;\n                });\n\n                return done(null, html);\n            }\n            let entry = cidList[pos++];\n            replaceCallback(entry.attachment, (err, url) => {\n                if (err) {\n                    return setImmediate(() => done(err));\n                }\n                entry.url = url;\n                setImmediate(processNext);\n            });\n        };\n\n        setImmediate(processNext);\n    }\n\n    textToHtml(str) {\n        if (this.options.skipTextToHtml) {\n            return '';\n        }\n        str = (str || '').toString();\n        let encoded;\n\n        let linkified = false;\n        if (!this.options.skipTextLinks) {\n            try {\n                if (linkify.pretest(str)) {\n                    linkified = true;\n                    let links = linkify.match(str) || [];\n                    let result = [];\n                    let last = 0;\n\n                    links.forEach(link => {\n                        if (last < link.index) {\n                            let textPart = he\n                                // encode special chars\n                                .encode(str.slice(last, link.index), {\n                                    useNamedReferences: true\n                                });\n                            result.push(textPart);\n                        }\n\n                        result.push(`<a href=\"${link.url}\">${link.text}</a>`);\n\n                        last = link.lastIndex;\n                    });\n\n                    let textPart = he\n                        // encode special chars\n                        .encode(str.slice(last), {\n                            useNamedReferences: true\n                        });\n                    result.push(textPart);\n\n                    encoded = result.join('');\n                }\n            } catch (E) {\n                // failed, don't linkify\n            }\n        }\n\n        if (!linkified) {\n            encoded = he\n                // encode special chars\n                .encode(str, {\n                    useNamedReferences: true\n                });\n        }\n\n        let text =\n            '<p>' +\n            encoded\n                .replace(/\\r?\\n/g, '\\n')\n                .trim() // normalize line endings\n                .replace(/[ \\t]+$/gm, '')\n                .trim() // trim empty line endings\n                .replace(/\\n\\n+/g, '</p><p>')\n                .trim() // insert <p> to multiple linebreaks\n                .replace(/\\n/g, '<br/>') + // insert <br> to single linebreaks\n            '</p>';\n\n        return text;\n    }\n}\n\nmodule.exports = MailParser;\n"],"names":[],"ignoreList":[0],"sourceRoot":""}\n//# sourceURL=webpack-internal:///(action-browser)/./node_modules/mailparser/lib/mail-parser.js\n");
|
|
|
|
/***/ }),
|
|
|
|
/***/ "(action-browser)/./node_modules/mailparser/lib/simple-parser.js":
|
|
/*!******************************************************!*\
|
|
!*** ./node_modules/mailparser/lib/simple-parser.js ***!
|
|
\******************************************************/
|
|
/***/ ((module, __unused_webpack_exports, __webpack_require__) => {
|
|
|
|
eval("\n\nconst MailParser = __webpack_require__(/*! ./mail-parser.js */ \"(action-browser)/./node_modules/mailparser/lib/mail-parser.js\");\n\nmodule.exports = (input, options, callback) => {\n if (input === null || input === undefined) {\n throw new TypeError('Input cannot be null or undefined.');\n }\n\n if (!callback && typeof options === 'function') {\n callback = options;\n options = false;\n }\n\n let promise;\n if (!callback) {\n promise = new Promise((resolve, reject) => {\n callback = callbackPromise(resolve, reject);\n });\n }\n\n options = options || {};\n let keepCidLinks = !!options.keepCidLinks;\n\n let mail = {\n attachments: []\n };\n\n let parser = new MailParser(options);\n\n parser.on('error', err => {\n callback(err);\n });\n\n parser.on('headers', headers => {\n mail.headers = headers;\n mail.headerLines = parser.headerLines;\n });\n\n let reading = false;\n let reader = () => {\n reading = true;\n\n let data = parser.read();\n\n if (data === null) {\n reading = false;\n return;\n }\n\n if (data.type === 'text') {\n Object.keys(data).forEach(key => {\n if (['text', 'html', 'textAsHtml'].includes(key)) {\n mail[key] = data[key];\n }\n });\n }\n\n if (data.type === 'attachment') {\n mail.attachments.push(data);\n\n let chunks = [];\n let chunklen = 0;\n data.content.on('readable', () => {\n let chunk;\n while ((chunk = data.content.read()) !== null) {\n chunks.push(chunk);\n chunklen += chunk.length;\n }\n });\n\n data.content.on('end', () => {\n data.content = Buffer.concat(chunks, chunklen);\n data.release();\n reader();\n });\n } else {\n reader();\n }\n };\n\n parser.on('readable', () => {\n if (!reading) {\n reader();\n }\n });\n\n parser.on('end', () => {\n ['subject', 'references', 'date', 'to', 'from', 'to', 'cc', 'bcc', 'message-id', 'in-reply-to', 'reply-to'].forEach(key => {\n if (mail.headers && mail.headers.has(key)) {\n mail[key.replace(/-([a-z])/g, (m, c) => c.toUpperCase())] = mail.headers.get(key);\n }\n });\n\n if (keepCidLinks) {\n return callback(null, mail);\n }\n parser.updateImageLinks(\n (attachment, done) => done(false, 'data:' + attachment.contentType + ';base64,' + attachment.content.toString('base64')),\n (err, html) => {\n if (err) {\n return callback(err);\n }\n mail.html = html;\n\n callback(null, mail);\n }\n );\n });\n\n if (typeof input === 'string') {\n parser.end(Buffer.from(input));\n } else if (Buffer.isBuffer(input)) {\n parser.end(input);\n } else {\n input\n .once('error', err => {\n input.destroy();\n parser.destroy();\n callback(err);\n })\n .pipe(parser);\n }\n\n return promise;\n};\n\nfunction callbackPromise(resolve, reject) {\n return function (...args) {\n let err = args.shift();\n if (err) {\n reject(err);\n } else {\n resolve(...args);\n }\n };\n}\n//# sourceURL=[module]\n//# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiKGFjdGlvbi1icm93c2VyKS8uL25vZGVfbW9kdWxlcy9tYWlscGFyc2VyL2xpYi9zaW1wbGUtcGFyc2VyLmpzIiwibWFwcGluZ3MiOiJBQUFhOztBQUViLG1CQUFtQixtQkFBTyxDQUFDLHVGQUFrQjs7QUFFN0M7QUFDQTtBQUNBO0FBQ0E7O0FBRUE7QUFDQTtBQUNBO0FBQ0E7O0FBRUE7QUFDQTtBQUNBO0FBQ0E7QUFDQSxTQUFTO0FBQ1Q7O0FBRUE7QUFDQTs7QUFFQTtBQUNBO0FBQ0E7O0FBRUE7O0FBRUE7QUFDQTtBQUNBLEtBQUs7O0FBRUw7QUFDQTtBQUNBO0FBQ0EsS0FBSzs7QUFFTDtBQUNBO0FBQ0E7O0FBRUE7O0FBRUE7QUFDQTtBQUNBO0FBQ0E7O0FBRUE7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBLGFBQWE7QUFDYjs7QUFFQTtBQUNBOztBQUVBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQSxhQUFhOztBQUViO0FBQ0E7QUFDQTtBQUNBO0FBQ0EsYUFBYTtBQUNiLFVBQVU7QUFDVjtBQUNBO0FBQ0E7O0FBRUE7QUFDQTtBQUNBO0FBQ0E7QUFDQSxLQUFLOztBQUVMO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQSxTQUFTOztBQUVUO0FBQ0E7QUFDQTtBQUNBO0FBQ0EsbUZBQW1GO0FBQ25GO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7O0FBRUE7QUFDQTtBQUNBO0FBQ0EsS0FBSzs7QUFFTDtBQUNBO0FBQ0EsTUFBTTtBQUNOO0FBQ0EsTUFBTTtBQUNOO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQSxhQUFhO0FBQ2I7QUFDQTs7QUFFQTtBQUNBOztBQUVBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQSxVQUFVO0FBQ1Y7QUFDQTtBQUNBO0FBQ0EiLCJzb3VyY2VzIjpbIi9ob21lL2FsbWEvbmV4dGdlbi9OZWFoLW1haWwvbm9kZV9tb2R1bGVzL21haWxwYXJzZXIvbGliL3NpbXBsZS1wYXJzZXIuanMiXSwic291cmNlc0NvbnRlbnQiOlsiJ3VzZSBzdHJpY3QnO1xuXG5jb25zdCBNYWlsUGFyc2VyID0gcmVxdWlyZSgnLi9tYWlsLXBhcnNlci5qcycpO1xuXG5tb2R1bGUuZXhwb3J0cyA9IChpbnB1dCwgb3B0aW9ucywgY2FsbGJhY2spID0+IHtcbiAgICBpZiAoaW5wdXQgPT09IG51bGwgfHwgaW5wdXQgPT09IHVuZGVmaW5lZCkge1xuICAgICAgICB0aHJvdyBuZXcgVHlwZUVycm9yKCdJbnB1dCBjYW5ub3QgYmUgbnVsbCBvciB1bmRlZmluZWQuJyk7XG4gICAgfVxuXG4gICAgaWYgKCFjYWxsYmFjayAmJiB0eXBlb2Ygb3B0aW9ucyA9PT0gJ2Z1bmN0aW9uJykge1xuICAgICAgICBjYWxsYmFjayA9IG9wdGlvbnM7XG4gICAgICAgIG9wdGlvbnMgPSBmYWxzZTtcbiAgICB9XG5cbiAgICBsZXQgcHJvbWlzZTtcbiAgICBpZiAoIWNhbGxiYWNrKSB7XG4gICAgICAgIHByb21pc2UgPSBuZXcgUHJvbWlzZSgocmVzb2x2ZSwgcmVqZWN0KSA9PiB7XG4gICAgICAgICAgICBjYWxsYmFjayA9IGNhbGxiYWNrUHJvbWlzZShyZXNvbHZlLCByZWplY3QpO1xuICAgICAgICB9KTtcbiAgICB9XG5cbiAgICBvcHRpb25zID0gb3B0aW9ucyB8fCB7fTtcbiAgICBsZXQga2VlcENpZExpbmtzID0gISFvcHRpb25zLmtlZXBDaWRMaW5rcztcblxuICAgIGxldCBtYWlsID0ge1xuICAgICAgICBhdHRhY2htZW50czogW11cbiAgICB9O1xuXG4gICAgbGV0IHBhcnNlciA9IG5ldyBNYWlsUGFyc2VyKG9wdGlvbnMpO1xuXG4gICAgcGFyc2VyLm9uKCdlcnJvcicsIGVyciA9PiB7XG4gICAgICAgIGNhbGxiYWNrKGVycik7XG4gICAgfSk7XG5cbiAgICBwYXJzZXIub24oJ2hlYWRlcnMnLCBoZWFkZXJzID0+IHtcbiAgICAgICAgbWFpbC5oZWFkZXJzID0gaGVhZGVycztcbiAgICAgICAgbWFpbC5oZWFkZXJMaW5lcyA9IHBhcnNlci5oZWFkZXJMaW5lcztcbiAgICB9KTtcblxuICAgIGxldCByZWFkaW5nID0gZmFsc2U7XG4gICAgbGV0IHJlYWRlciA9ICgpID0+IHtcbiAgICAgICAgcmVhZGluZyA9IHRydWU7XG5cbiAgICAgICAgbGV0IGRhdGEgPSBwYXJzZXIucmVhZCgpO1xuXG4gICAgICAgIGlmIChkYXRhID09PSBudWxsKSB7XG4gICAgICAgICAgICByZWFkaW5nID0gZmFsc2U7XG4gICAgICAgICAgICByZXR1cm47XG4gICAgICAgIH1cblxuICAgICAgICBpZiAoZGF0YS50eXBlID09PSAndGV4dCcpIHtcbiAgICAgICAgICAgIE9iamVjdC5rZXlzKGRhdGEpLmZvckVhY2goa2V5ID0+IHtcbiAgICAgICAgICAgICAgICBpZiAoWyd0ZXh0JywgJ2h0bWwnLCAndGV4dEFzSHRtbCddLmluY2x1ZGVzKGtleSkpIHtcbiAgICAgICAgICAgICAgICAgICAgbWFpbFtrZXldID0gZGF0YVtrZXldO1xuICAgICAgICAgICAgICAgIH1cbiAgICAgICAgICAgIH0pO1xuICAgICAgICB9XG5cbiAgICAgICAgaWYgKGRhdGEudHlwZSA9PT0gJ2F0dGFjaG1lbnQnKSB7XG4gICAgICAgICAgICBtYWlsLmF0dGFjaG1lbnRzLnB1c2goZGF0YSk7XG5cbiAgICAgICAgICAgIGxldCBjaHVua3MgPSBbXTtcbiAgICAgICAgICAgIGxldCBjaHVua2xlbiA9IDA7XG4gICAgICAgICAgICBkYXRhLmNvbnRlbnQub24oJ3JlYWRhYmxlJywgKCkgPT4ge1xuICAgICAgICAgICAgICAgIGxldCBjaHVuaztcbiAgICAgICAgICAgICAgICB3aGlsZSAoKGNodW5rID0gZGF0YS5jb250ZW50LnJlYWQoKSkgIT09IG51bGwpIHtcbiAgICAgICAgICAgICAgICAgICAgY2h1bmtzLnB1c2goY2h1bmspO1xuICAgICAgICAgICAgICAgICAgICBjaHVua2xlbiArPSBjaHVuay5sZW5ndGg7XG4gICAgICAgICAgICAgICAgfVxuICAgICAgICAgICAgfSk7XG5cbiAgICAgICAgICAgIGRhdGEuY29udGVudC5vbignZW5kJywgKCkgPT4ge1xuICAgICAgICAgICAgICAgIGRhdGEuY29udGVudCA9IEJ1ZmZlci5jb25jYXQoY2h1bmtzLCBjaHVua2xlbik7XG4gICAgICAgICAgICAgICAgZGF0YS5yZWxlYXNlKCk7XG4gICAgICAgICAgICAgICAgcmVhZGVyKCk7XG4gICAgICAgICAgICB9KTtcbiAgICAgICAgfSBlbHNlIHtcbiAgICAgICAgICAgIHJlYWRlcigpO1xuICAgICAgICB9XG4gICAgfTtcblxuICAgIHBhcnNlci5vbigncmVhZGFibGUnLCAoKSA9PiB7XG4gICAgICAgIGlmICghcmVhZGluZykge1xuICAgICAgICAgICAgcmVhZGVyKCk7XG4gICAgICAgIH1cbiAgICB9KTtcblxuICAgIHBhcnNlci5vbignZW5kJywgKCkgPT4ge1xuICAgICAgICBbJ3N1YmplY3QnLCAncmVmZXJlbmNlcycsICdkYXRlJywgJ3RvJywgJ2Zyb20nLCAndG8nLCAnY2MnLCAnYmNjJywgJ21lc3NhZ2UtaWQnLCAnaW4tcmVwbHktdG8nLCAncmVwbHktdG8nXS5mb3JFYWNoKGtleSA9PiB7XG4gICAgICAgICAgICBpZiAobWFpbC5oZWFkZXJzICYmIG1haWwuaGVhZGVycy5oYXMoa2V5KSkge1xuICAgICAgICAgICAgICAgIG1haWxba2V5LnJlcGxhY2UoLy0oW2Etel0pL2csIChtLCBjKSA9PiBjLnRvVXBwZXJDYXNlKCkpXSA9IG1haWwuaGVhZGVycy5nZXQoa2V5KTtcbiAgICAgICAgICAgIH1cbiAgICAgICAgfSk7XG5cbiAgICAgICAgaWYgKGtlZXBDaWRMaW5rcykge1xuICAgICAgICAgICAgcmV0dXJuIGNhbGxiYWNrKG51bGwsIG1haWwpO1xuICAgICAgICB9XG4gICAgICAgIHBhcnNlci51cGRhdGVJbWFnZUxpbmtzKFxuICAgICAgICAgICAgKGF0dGFjaG1lbnQsIGRvbmUpID0+IGRvbmUoZmFsc2UsICdkYXRhOicgKyBhdHRhY2htZW50LmNvbnRlbnRUeXBlICsgJztiYXNlNjQsJyArIGF0dGFjaG1lbnQuY29udGVudC50b1N0cmluZygnYmFzZTY0JykpLFxuICAgICAgICAgICAgKGVyciwgaHRtbCkgPT4ge1xuICAgICAgICAgICAgICAgIGlmIChlcnIpIHtcbiAgICAgICAgICAgICAgICAgICAgcmV0dXJuIGNhbGxiYWNrKGVycik7XG4gICAgICAgICAgICAgICAgfVxuICAgICAgICAgICAgICAgIG1haWwuaHRtbCA9IGh0bWw7XG5cbiAgICAgICAgICAgICAgICBjYWxsYmFjayhudWxsLCBtYWlsKTtcbiAgICAgICAgICAgIH1cbiAgICAgICAgKTtcbiAgICB9KTtcblxuICAgIGlmICh0eXBlb2YgaW5wdXQgPT09ICdzdHJpbmcnKSB7XG4gICAgICAgIHBhcnNlci5lbmQoQnVmZmVyLmZyb20oaW5wdXQpKTtcbiAgICB9IGVsc2UgaWYgKEJ1ZmZlci5pc0J1ZmZlcihpbnB1dCkpIHtcbiAgICAgICAgcGFyc2VyLmVuZChpbnB1dCk7XG4gICAgfSBlbHNlIHtcbiAgICAgICAgaW5wdXRcbiAgICAgICAgICAgIC5vbmNlKCdlcnJvcicsIGVyciA9PiB7XG4gICAgICAgICAgICAgICAgaW5wdXQuZGVzdHJveSgpO1xuICAgICAgICAgICAgICAgIHBhcnNlci5kZXN0cm95KCk7XG4gICAgICAgICAgICAgICAgY2FsbGJhY2soZXJyKTtcbiAgICAgICAgICAgIH0pXG4gICAgICAgICAgICAucGlwZShwYXJzZXIpO1xuICAgIH1cblxuICAgIHJldHVybiBwcm9taXNlO1xufTtcblxuZnVuY3Rpb24gY2FsbGJhY2tQcm9taXNlKHJlc29sdmUsIHJlamVjdCkge1xuICAgIHJldHVybiBmdW5jdGlvbiAoLi4uYXJncykge1xuICAgICAgICBsZXQgZXJyID0gYXJncy5zaGlmdCgpO1xuICAgICAgICBpZiAoZXJyKSB7XG4gICAgICAgICAgICByZWplY3QoZXJyKTtcbiAgICAgICAgfSBlbHNlIHtcbiAgICAgICAgICAgIHJlc29sdmUoLi4uYXJncyk7XG4gICAgICAgIH1cbiAgICB9O1xufVxuIl0sIm5hbWVzIjpbXSwiaWdub3JlTGlzdCI6WzBdLCJzb3VyY2VSb290IjoiIn0=\n//# sourceURL=webpack-internal:///(action-browser)/./node_modules/mailparser/lib/simple-parser.js\n");
|
|
|
|
/***/ }),
|
|
|
|
/***/ "(action-browser)/./node_modules/mailparser/lib/stream-hash.js":
|
|
/*!****************************************************!*\
|
|
!*** ./node_modules/mailparser/lib/stream-hash.js ***!
|
|
\****************************************************/
|
|
/***/ ((module, __unused_webpack_exports, __webpack_require__) => {
|
|
|
|
eval("\n\nconst crypto = __webpack_require__(/*! crypto */ \"crypto\");\nconst Transform = (__webpack_require__(/*! stream */ \"stream\").Transform);\n\nclass StreamHash extends Transform {\n constructor(attachment, algo) {\n super();\n this.attachment = attachment;\n this.algo = (algo || 'md5').toLowerCase();\n this.hash = crypto.createHash(algo);\n this.byteCount = 0;\n }\n\n _transform(chunk, encoding, done) {\n this.hash.update(chunk);\n this.byteCount += chunk.length;\n done(null, chunk);\n }\n\n _flush(done) {\n this.attachment.checksum = this.hash.digest('hex');\n this.attachment.size = this.byteCount;\n done();\n }\n}\n\nmodule.exports = StreamHash;\n//# sourceURL=[module]\n//# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiKGFjdGlvbi1icm93c2VyKS8uL25vZGVfbW9kdWxlcy9tYWlscGFyc2VyL2xpYi9zdHJlYW0taGFzaC5qcyIsIm1hcHBpbmdzIjoiQUFBYTs7QUFFYixlQUFlLG1CQUFPLENBQUMsc0JBQVE7QUFDL0Isa0JBQWtCLHVEQUEyQjs7QUFFN0M7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTs7QUFFQTtBQUNBO0FBQ0E7QUFDQTtBQUNBOztBQUVBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTs7QUFFQSIsInNvdXJjZXMiOlsiL2hvbWUvYWxtYS9uZXh0Z2VuL05lYWgtbWFpbC9ub2RlX21vZHVsZXMvbWFpbHBhcnNlci9saWIvc3RyZWFtLWhhc2guanMiXSwic291cmNlc0NvbnRlbnQiOlsiJ3VzZSBzdHJpY3QnO1xuXG5jb25zdCBjcnlwdG8gPSByZXF1aXJlKCdjcnlwdG8nKTtcbmNvbnN0IFRyYW5zZm9ybSA9IHJlcXVpcmUoJ3N0cmVhbScpLlRyYW5zZm9ybTtcblxuY2xhc3MgU3RyZWFtSGFzaCBleHRlbmRzIFRyYW5zZm9ybSB7XG4gICAgY29uc3RydWN0b3IoYXR0YWNobWVudCwgYWxnbykge1xuICAgICAgICBzdXBlcigpO1xuICAgICAgICB0aGlzLmF0dGFjaG1lbnQgPSBhdHRhY2htZW50O1xuICAgICAgICB0aGlzLmFsZ28gPSAoYWxnbyB8fCAnbWQ1JykudG9Mb3dlckNhc2UoKTtcbiAgICAgICAgdGhpcy5oYXNoID0gY3J5cHRvLmNyZWF0ZUhhc2goYWxnbyk7XG4gICAgICAgIHRoaXMuYnl0ZUNvdW50ID0gMDtcbiAgICB9XG5cbiAgICBfdHJhbnNmb3JtKGNodW5rLCBlbmNvZGluZywgZG9uZSkge1xuICAgICAgICB0aGlzLmhhc2gudXBkYXRlKGNodW5rKTtcbiAgICAgICAgdGhpcy5ieXRlQ291bnQgKz0gY2h1bmsubGVuZ3RoO1xuICAgICAgICBkb25lKG51bGwsIGNodW5rKTtcbiAgICB9XG5cbiAgICBfZmx1c2goZG9uZSkge1xuICAgICAgICB0aGlzLmF0dGFjaG1lbnQuY2hlY2tzdW0gPSB0aGlzLmhhc2guZGlnZXN0KCdoZXgnKTtcbiAgICAgICAgdGhpcy5hdHRhY2htZW50LnNpemUgPSB0aGlzLmJ5dGVDb3VudDtcbiAgICAgICAgZG9uZSgpO1xuICAgIH1cbn1cblxubW9kdWxlLmV4cG9ydHMgPSBTdHJlYW1IYXNoO1xuIl0sIm5hbWVzIjpbXSwiaWdub3JlTGlzdCI6WzBdLCJzb3VyY2VSb290IjoiIn0=\n//# sourceURL=webpack-internal:///(action-browser)/./node_modules/mailparser/lib/stream-hash.js\n");
|
|
|
|
/***/ }),
|
|
|
|
/***/ "(action-browser)/./node_modules/mailparser/node_modules/nodemailer/lib/addressparser/index.js":
|
|
/*!************************************************************************************!*\
|
|
!*** ./node_modules/mailparser/node_modules/nodemailer/lib/addressparser/index.js ***!
|
|
\************************************************************************************/
|
|
/***/ ((module) => {
|
|
|
|
eval("\n\n/**\n * Converts tokens for a single address into an address object\n *\n * @param {Array} tokens Tokens object\n * @return {Object} Address object\n */\nfunction _handleAddress(tokens) {\n let isGroup = false;\n let state = 'text';\n let address;\n let addresses = [];\n let data = {\n address: [],\n comment: [],\n group: [],\n text: []\n };\n let i;\n let len;\n\n // Filter out <addresses>, (comments) and regular text\n for (i = 0, len = tokens.length; i < len; i++) {\n let token = tokens[i];\n let prevToken = i ? tokens[i - 1] : null;\n if (token.type === 'operator') {\n switch (token.value) {\n case '<':\n state = 'address';\n break;\n case '(':\n state = 'comment';\n break;\n case ':':\n state = 'group';\n isGroup = true;\n break;\n default:\n state = 'text';\n break;\n }\n } else if (token.value) {\n if (state === 'address') {\n // handle use case where unquoted name includes a \"<\"\n // Apple Mail truncates everything between an unexpected < and an address\n // and so will we\n token.value = token.value.replace(/^[^<]*<\\s*/, '');\n }\n\n if (prevToken && prevToken.noBreak && data[state].length) {\n // join values\n data[state][data[state].length - 1] += token.value;\n } else {\n data[state].push(token.value);\n }\n }\n }\n\n // If there is no text but a comment, replace the two\n if (!data.text.length && data.comment.length) {\n data.text = data.comment;\n data.comment = [];\n }\n\n if (isGroup) {\n // http://tools.ietf.org/html/rfc2822#appendix-A.1.3\n data.text = data.text.join(' ');\n addresses.push({\n name: data.text || (address && address.name),\n group: data.group.length ? addressparser(data.group.join(',')) : []\n });\n } else {\n // If no address was found, try to detect one from regular text\n if (!data.address.length && data.text.length) {\n for (i = data.text.length - 1; i >= 0; i--) {\n if (data.text[i].match(/^[^@\\s]+@[^@\\s]+$/)) {\n data.address = data.text.splice(i, 1);\n break;\n }\n }\n\n let _regexHandler = function (address) {\n if (!data.address.length) {\n data.address = [address.trim()];\n return ' ';\n } else {\n return address;\n }\n };\n\n // still no address\n if (!data.address.length) {\n for (i = data.text.length - 1; i >= 0; i--) {\n // fixed the regex to parse email address correctly when email address has more than one @\n data.text[i] = data.text[i].replace(/\\s*\\b[^@\\s]+@[^\\s]+\\b\\s*/, _regexHandler).trim();\n if (data.address.length) {\n break;\n }\n }\n }\n }\n\n // If there's still is no text but a comment exixts, replace the two\n if (!data.text.length && data.comment.length) {\n data.text = data.comment;\n data.comment = [];\n }\n\n // Keep only the first address occurence, push others to regular text\n if (data.address.length > 1) {\n data.text = data.text.concat(data.address.splice(1));\n }\n\n // Join values with spaces\n data.text = data.text.join(' ');\n data.address = data.address.join(' ');\n\n if (!data.address && isGroup) {\n return [];\n } else {\n address = {\n address: data.address || data.text || '',\n name: data.text || data.address || ''\n };\n\n if (address.address === address.name) {\n if ((address.address || '').match(/@/)) {\n address.name = '';\n } else {\n address.address = '';\n }\n }\n\n addresses.push(address);\n }\n }\n\n return addresses;\n}\n\n/**\n * Creates a Tokenizer object for tokenizing address field strings\n *\n * @constructor\n * @param {String} str Address field string\n */\nclass Tokenizer {\n constructor(str) {\n this.str = (str || '').toString();\n this.operatorCurrent = '';\n this.operatorExpecting = '';\n this.node = null;\n this.escaped = false;\n\n this.list = [];\n /**\n * Operator tokens and which tokens are expected to end the sequence\n */\n this.operators = {\n '\"': '\"',\n '(': ')',\n '<': '>',\n ',': '',\n ':': ';',\n // Semicolons are not a legal delimiter per the RFC2822 grammar other\n // than for terminating a group, but they are also not valid for any\n // other use in this context. Given that some mail clients have\n // historically allowed the semicolon as a delimiter equivalent to the\n // comma in their UI, it makes sense to treat them the same as a comma\n // when used outside of a group.\n ';': ''\n };\n }\n\n /**\n * Tokenizes the original input string\n *\n * @return {Array} An array of operator|text tokens\n */\n tokenize() {\n let list = [];\n\n for (let i = 0, len = this.str.length; i < len; i++) {\n let chr = this.str.charAt(i);\n let nextChr = i < len - 1 ? this.str.charAt(i + 1) : null;\n this.checkChar(chr, nextChr);\n }\n\n this.list.forEach(node => {\n node.value = (node.value || '').toString().trim();\n if (node.value) {\n list.push(node);\n }\n });\n\n return list;\n }\n\n /**\n * Checks if a character is an operator or text and acts accordingly\n *\n * @param {String} chr Character from the address field\n */\n checkChar(chr, nextChr) {\n if (this.escaped) {\n // ignore next condition blocks\n } else if (chr === this.operatorExpecting) {\n this.node = {\n type: 'operator',\n value: chr\n };\n\n if (nextChr && ![' ', '\\t', '\\r', '\\n', ',', ';'].includes(nextChr)) {\n this.node.noBreak = true;\n }\n\n this.list.push(this.node);\n this.node = null;\n this.operatorExpecting = '';\n this.escaped = false;\n\n return;\n } else if (!this.operatorExpecting && chr in this.operators) {\n this.node = {\n type: 'operator',\n value: chr\n };\n this.list.push(this.node);\n this.node = null;\n this.operatorExpecting = this.operators[chr];\n this.escaped = false;\n return;\n } else if (['\"', \"'\"].includes(this.operatorExpecting) && chr === '\\\\') {\n this.escaped = true;\n return;\n }\n\n if (!this.node) {\n this.node = {\n type: 'text',\n value: ''\n };\n this.list.push(this.node);\n }\n\n if (chr === '\\n') {\n // Convert newlines to spaces. Carriage return is ignored as \\r and \\n usually\n // go together anyway and there already is a WS for \\n. Lone \\r means something is fishy.\n chr = ' ';\n }\n\n if (chr.charCodeAt(0) >= 0x21 || [' ', '\\t'].includes(chr)) {\n // skip command bytes\n this.node.value += chr;\n }\n\n this.escaped = false;\n }\n}\n\n/**\n * Parses structured e-mail addresses from an address field\n *\n * Example:\n *\n * 'Name <address@domain>'\n *\n * will be converted to\n *\n * [{name: 'Name', address: 'address@domain'}]\n *\n * @param {String} str Address field\n * @return {Array} An array of address objects\n */\nfunction addressparser(str, options) {\n options = options || {};\n\n let tokenizer = new Tokenizer(str);\n let tokens = tokenizer.tokenize();\n\n let addresses = [];\n let address = [];\n let parsedAddresses = [];\n\n tokens.forEach(token => {\n if (token.type === 'operator' && (token.value === ',' || token.value === ';')) {\n if (address.length) {\n addresses.push(address);\n }\n address = [];\n } else {\n address.push(token);\n }\n });\n\n if (address.length) {\n addresses.push(address);\n }\n\n addresses.forEach(address => {\n address = _handleAddress(address);\n if (address.length) {\n parsedAddresses = parsedAddresses.concat(address);\n }\n });\n\n if (options.flatten) {\n let addresses = [];\n let walkAddressList = list => {\n list.forEach(address => {\n if (address.group) {\n return walkAddressList(address.group);\n } else {\n addresses.push(address);\n }\n });\n };\n walkAddressList(parsedAddresses);\n return addresses;\n }\n\n return parsedAddresses;\n}\n\n// expose to the world\nmodule.exports = addressparser;\n//# sourceURL=[module]\n//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"file":"(action-browser)/./node_modules/mailparser/node_modules/nodemailer/lib/addressparser/index.js","mappings":"AAAa;;AAEb;AACA;AACA;AACA,WAAW,OAAO;AAClB,YAAY,QAAQ;AACpB;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA,qCAAqC,SAAS;AAC9C;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,UAAU;AACV;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA,cAAc;AACd;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA,SAAS;AACT,MAAM;AACN;AACA;AACA,2CAA2C,QAAQ;AACnD;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA,kBAAkB;AAClB;AACA;AACA;;AAEA;AACA;AACA,+CAA+C,QAAQ;AACvD;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;;AAEA;AACA;AACA;;AAEA;AACA;AACA,UAAU;AACV;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA,kBAAkB;AAClB;AACA;AACA;;AAEA;AACA;AACA;;AAEA;AACA;;AAEA;AACA;AACA;AACA;AACA,WAAW,QAAQ;AACnB;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,mBAAmB;AACnB;AACA;AACA;AACA;AACA;AACA;AACA,cAAc;AACd;AACA;;AAEA;AACA;AACA;AACA,gBAAgB,OAAO;AACvB;AACA;AACA;;AAEA,+CAA+C,SAAS;AACxD;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA,SAAS;;AAET;AACA;;AAEA;AACA;AACA;AACA,eAAe,QAAQ;AACvB;AACA;AACA;AACA;AACA,UAAU;AACV;AACA;AACA;AACA;;AAEA,2DAA2D;AAC3D;AACA;;AAEA;AACA;AACA;AACA;;AAEA;AACA,UAAU;AACV;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,UAAU;AACV;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;;AAEA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,SAAS,wCAAwC;AACjD;AACA,WAAW,QAAQ;AACnB,YAAY,OAAO;AACnB;AACA;AACA;;AAEA;AACA;;AAEA;AACA;AACA;;AAEA;AACA,mFAAmF;AACnF;AACA;AACA;AACA;AACA,UAAU;AACV;AACA;AACA,KAAK;;AAEL;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA,KAAK;;AAEL;AACA;AACA;AACA;AACA;AACA;AACA,kBAAkB;AAClB;AACA;AACA,aAAa;AACb;AACA;AACA;AACA;;AAEA;AACA;;AAEA;AACA","sources":["/home/alma/nextgen/Neah-mail/node_modules/mailparser/node_modules/nodemailer/lib/addressparser/index.js"],"sourcesContent":["'use strict';\n\n/**\n * Converts tokens for a single address into an address object\n *\n * @param {Array} tokens Tokens object\n * @return {Object} Address object\n */\nfunction _handleAddress(tokens) {\n    let isGroup = false;\n    let state = 'text';\n    let address;\n    let addresses = [];\n    let data = {\n        address: [],\n        comment: [],\n        group: [],\n        text: []\n    };\n    let i;\n    let len;\n\n    // Filter out <addresses>, (comments) and regular text\n    for (i = 0, len = tokens.length; i < len; i++) {\n        let token = tokens[i];\n        let prevToken = i ? tokens[i - 1] : null;\n        if (token.type === 'operator') {\n            switch (token.value) {\n                case '<':\n                    state = 'address';\n                    break;\n                case '(':\n                    state = 'comment';\n                    break;\n                case ':':\n                    state = 'group';\n                    isGroup = true;\n                    break;\n                default:\n                    state = 'text';\n                    break;\n            }\n        } else if (token.value) {\n            if (state === 'address') {\n                // handle use case where unquoted name includes a \"<\"\n                // Apple Mail truncates everything between an unexpected < and an address\n                // and so will we\n                token.value = token.value.replace(/^[^<]*<\\s*/, '');\n            }\n\n            if (prevToken && prevToken.noBreak && data[state].length) {\n                // join values\n                data[state][data[state].length - 1] += token.value;\n            } else {\n                data[state].push(token.value);\n            }\n        }\n    }\n\n    // If there is no text but a comment, replace the two\n    if (!data.text.length && data.comment.length) {\n        data.text = data.comment;\n        data.comment = [];\n    }\n\n    if (isGroup) {\n        // http://tools.ietf.org/html/rfc2822#appendix-A.1.3\n        data.text = data.text.join(' ');\n        addresses.push({\n            name: data.text || (address && address.name),\n            group: data.group.length ? addressparser(data.group.join(',')) : []\n        });\n    } else {\n        // If no address was found, try to detect one from regular text\n        if (!data.address.length && data.text.length) {\n            for (i = data.text.length - 1; i >= 0; i--) {\n                if (data.text[i].match(/^[^@\\s]+@[^@\\s]+$/)) {\n                    data.address = data.text.splice(i, 1);\n                    break;\n                }\n            }\n\n            let _regexHandler = function (address) {\n                if (!data.address.length) {\n                    data.address = [address.trim()];\n                    return ' ';\n                } else {\n                    return address;\n                }\n            };\n\n            // still no address\n            if (!data.address.length) {\n                for (i = data.text.length - 1; i >= 0; i--) {\n                    // fixed the regex to parse email address correctly when email address has more than one @\n                    data.text[i] = data.text[i].replace(/\\s*\\b[^@\\s]+@[^\\s]+\\b\\s*/, _regexHandler).trim();\n                    if (data.address.length) {\n                        break;\n                    }\n                }\n            }\n        }\n\n        // If there's still is no text but a comment exixts, replace the two\n        if (!data.text.length && data.comment.length) {\n            data.text = data.comment;\n            data.comment = [];\n        }\n\n        // Keep only the first address occurence, push others to regular text\n        if (data.address.length > 1) {\n            data.text = data.text.concat(data.address.splice(1));\n        }\n\n        // Join values with spaces\n        data.text = data.text.join(' ');\n        data.address = data.address.join(' ');\n\n        if (!data.address && isGroup) {\n            return [];\n        } else {\n            address = {\n                address: data.address || data.text || '',\n                name: data.text || data.address || ''\n            };\n\n            if (address.address === address.name) {\n                if ((address.address || '').match(/@/)) {\n                    address.name = '';\n                } else {\n                    address.address = '';\n                }\n            }\n\n            addresses.push(address);\n        }\n    }\n\n    return addresses;\n}\n\n/**\n * Creates a Tokenizer object for tokenizing address field strings\n *\n * @constructor\n * @param {String} str Address field string\n */\nclass Tokenizer {\n    constructor(str) {\n        this.str = (str || '').toString();\n        this.operatorCurrent = '';\n        this.operatorExpecting = '';\n        this.node = null;\n        this.escaped = false;\n\n        this.list = [];\n        /**\n         * Operator tokens and which tokens are expected to end the sequence\n         */\n        this.operators = {\n            '\"': '\"',\n            '(': ')',\n            '<': '>',\n            ',': '',\n            ':': ';',\n            // Semicolons are not a legal delimiter per the RFC2822 grammar other\n            // than for terminating a group, but they are also not valid for any\n            // other use in this context.  Given that some mail clients have\n            // historically allowed the semicolon as a delimiter equivalent to the\n            // comma in their UI, it makes sense to treat them the same as a comma\n            // when used outside of a group.\n            ';': ''\n        };\n    }\n\n    /**\n     * Tokenizes the original input string\n     *\n     * @return {Array} An array of operator|text tokens\n     */\n    tokenize() {\n        let list = [];\n\n        for (let i = 0, len = this.str.length; i < len; i++) {\n            let chr = this.str.charAt(i);\n            let nextChr = i < len - 1 ? this.str.charAt(i + 1) : null;\n            this.checkChar(chr, nextChr);\n        }\n\n        this.list.forEach(node => {\n            node.value = (node.value || '').toString().trim();\n            if (node.value) {\n                list.push(node);\n            }\n        });\n\n        return list;\n    }\n\n    /**\n     * Checks if a character is an operator or text and acts accordingly\n     *\n     * @param {String} chr Character from the address field\n     */\n    checkChar(chr, nextChr) {\n        if (this.escaped) {\n            // ignore next condition blocks\n        } else if (chr === this.operatorExpecting) {\n            this.node = {\n                type: 'operator',\n                value: chr\n            };\n\n            if (nextChr && ![' ', '\\t', '\\r', '\\n', ',', ';'].includes(nextChr)) {\n                this.node.noBreak = true;\n            }\n\n            this.list.push(this.node);\n            this.node = null;\n            this.operatorExpecting = '';\n            this.escaped = false;\n\n            return;\n        } else if (!this.operatorExpecting && chr in this.operators) {\n            this.node = {\n                type: 'operator',\n                value: chr\n            };\n            this.list.push(this.node);\n            this.node = null;\n            this.operatorExpecting = this.operators[chr];\n            this.escaped = false;\n            return;\n        } else if (['\"', \"'\"].includes(this.operatorExpecting) && chr === '\\\\') {\n            this.escaped = true;\n            return;\n        }\n\n        if (!this.node) {\n            this.node = {\n                type: 'text',\n                value: ''\n            };\n            this.list.push(this.node);\n        }\n\n        if (chr === '\\n') {\n            // Convert newlines to spaces. Carriage return is ignored as \\r and \\n usually\n            // go together anyway and there already is a WS for \\n. Lone \\r means something is fishy.\n            chr = ' ';\n        }\n\n        if (chr.charCodeAt(0) >= 0x21 || [' ', '\\t'].includes(chr)) {\n            // skip command bytes\n            this.node.value += chr;\n        }\n\n        this.escaped = false;\n    }\n}\n\n/**\n * Parses structured e-mail addresses from an address field\n *\n * Example:\n *\n *    'Name <address@domain>'\n *\n * will be converted to\n *\n *     [{name: 'Name', address: 'address@domain'}]\n *\n * @param {String} str Address field\n * @return {Array} An array of address objects\n */\nfunction addressparser(str, options) {\n    options = options || {};\n\n    let tokenizer = new Tokenizer(str);\n    let tokens = tokenizer.tokenize();\n\n    let addresses = [];\n    let address = [];\n    let parsedAddresses = [];\n\n    tokens.forEach(token => {\n        if (token.type === 'operator' && (token.value === ',' || token.value === ';')) {\n            if (address.length) {\n                addresses.push(address);\n            }\n            address = [];\n        } else {\n            address.push(token);\n        }\n    });\n\n    if (address.length) {\n        addresses.push(address);\n    }\n\n    addresses.forEach(address => {\n        address = _handleAddress(address);\n        if (address.length) {\n            parsedAddresses = parsedAddresses.concat(address);\n        }\n    });\n\n    if (options.flatten) {\n        let addresses = [];\n        let walkAddressList = list => {\n            list.forEach(address => {\n                if (address.group) {\n                    return walkAddressList(address.group);\n                } else {\n                    addresses.push(address);\n                }\n            });\n        };\n        walkAddressList(parsedAddresses);\n        return addresses;\n    }\n\n    return parsedAddresses;\n}\n\n// expose to the world\nmodule.exports = addressparser;\n"],"names":[],"ignoreList":[0],"sourceRoot":""}\n//# sourceURL=webpack-internal:///(action-browser)/./node_modules/mailparser/node_modules/nodemailer/lib/addressparser/index.js\n");
|
|
|
|
/***/ }),
|
|
|
|
/***/ "(rsc)/./node_modules/mailparser/index.js":
|
|
/*!******************************************!*\
|
|
!*** ./node_modules/mailparser/index.js ***!
|
|
\******************************************/
|
|
/***/ ((module, __unused_webpack_exports, __webpack_require__) => {
|
|
|
|
eval("\n\nconst MailParser = __webpack_require__(/*! ./lib/mail-parser */ \"(rsc)/./node_modules/mailparser/lib/mail-parser.js\");\nconst simpleParser = __webpack_require__(/*! ./lib/simple-parser */ \"(rsc)/./node_modules/mailparser/lib/simple-parser.js\");\n\nmodule.exports = {\n MailParser,\n simpleParser\n};\n//# sourceURL=[module]\n//# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiKHJzYykvLi9ub2RlX21vZHVsZXMvbWFpbHBhcnNlci9pbmRleC5qcyIsIm1hcHBpbmdzIjoiQUFBYTs7QUFFYixtQkFBbUIsbUJBQU8sQ0FBQyw2RUFBbUI7QUFDOUMscUJBQXFCLG1CQUFPLENBQUMsaUZBQXFCOztBQUVsRDtBQUNBO0FBQ0E7QUFDQSIsInNvdXJjZXMiOlsiL2hvbWUvYWxtYS9uZXh0Z2VuL05lYWgtbWFpbC9ub2RlX21vZHVsZXMvbWFpbHBhcnNlci9pbmRleC5qcyJdLCJzb3VyY2VzQ29udGVudCI6WyIndXNlIHN0cmljdCc7XG5cbmNvbnN0IE1haWxQYXJzZXIgPSByZXF1aXJlKCcuL2xpYi9tYWlsLXBhcnNlcicpO1xuY29uc3Qgc2ltcGxlUGFyc2VyID0gcmVxdWlyZSgnLi9saWIvc2ltcGxlLXBhcnNlcicpO1xuXG5tb2R1bGUuZXhwb3J0cyA9IHtcbiAgICBNYWlsUGFyc2VyLFxuICAgIHNpbXBsZVBhcnNlclxufTtcbiJdLCJuYW1lcyI6W10sImlnbm9yZUxpc3QiOlswXSwic291cmNlUm9vdCI6IiJ9\n//# sourceURL=webpack-internal:///(rsc)/./node_modules/mailparser/index.js\n");
|
|
|
|
/***/ }),
|
|
|
|
/***/ "(rsc)/./node_modules/mailparser/lib/mail-parser.js":
|
|
/*!****************************************************!*\
|
|
!*** ./node_modules/mailparser/lib/mail-parser.js ***!
|
|
\****************************************************/
|
|
/***/ ((module, __unused_webpack_exports, __webpack_require__) => {
|
|
|
|
eval("\n\nconst mailsplit = __webpack_require__(/*! mailsplit */ \"(rsc)/./node_modules/mailsplit/index.js\");\nconst libmime = __webpack_require__(/*! libmime */ \"(rsc)/./node_modules/libmime/lib/libmime.js\");\nconst addressparser = __webpack_require__(/*! nodemailer/lib/addressparser */ \"(rsc)/./node_modules/mailparser/node_modules/nodemailer/lib/addressparser/index.js\");\nconst Transform = (__webpack_require__(/*! stream */ \"stream\").Transform);\nconst Splitter = mailsplit.Splitter;\nconst punycode = __webpack_require__(/*! punycode.js */ \"(rsc)/./node_modules/punycode.js/punycode.es6.js\");\nconst FlowedDecoder = __webpack_require__(/*! mailsplit/lib/flowed-decoder */ \"(rsc)/./node_modules/mailsplit/lib/flowed-decoder.js\");\nconst StreamHash = __webpack_require__(/*! ./stream-hash */ \"(rsc)/./node_modules/mailparser/lib/stream-hash.js\");\nconst iconv = __webpack_require__(/*! iconv-lite */ \"(rsc)/./node_modules/iconv-lite/lib/index.js\");\nconst { htmlToText } = __webpack_require__(/*! html-to-text */ \"(rsc)/./node_modules/html-to-text/lib/html-to-text.cjs\");\nconst he = __webpack_require__(/*! he */ \"(rsc)/./node_modules/he/he.js\");\nconst linkify = __webpack_require__(/*! linkify-it */ \"(rsc)/./node_modules/linkify-it/build/index.cjs.js\")();\nconst tlds = __webpack_require__(/*! tlds */ \"(rsc)/./node_modules/tlds/index.json\");\nconst encodingJapanese = __webpack_require__(/*! encoding-japanese */ \"(rsc)/./node_modules/encoding-japanese/src/index.js\");\n\nlinkify\n .tlds(tlds) // Reload with full tlds list\n .tlds('onion', true) // Add unofficial `.onion` domain\n .add('git:', 'http:') // Add `git:` ptotocol as \"alias\"\n .add('ftp:', null) // Disable `ftp:` ptotocol\n .set({ fuzzyIP: true, fuzzyLink: true, fuzzyEmail: true });\n\n// twitter linkifier from\n// https://github.com/markdown-it/linkify-it#example-2-add-twitter-mentions-handler\nlinkify.add('@', {\n validate(text, pos, self) {\n let tail = text.slice(pos);\n\n if (!self.re.twitter) {\n self.re.twitter = new RegExp('^([a-zA-Z0-9_]){1,15}(?!_)(?=$|' + self.re.src_ZPCc + ')');\n }\n if (self.re.twitter.test(tail)) {\n // Linkifier allows punctuation chars before prefix,\n // but we additionally disable `@` (\"@@mention\" is invalid)\n if (pos >= 2 && tail[pos - 2] === '@') {\n return false;\n }\n return tail.match(self.re.twitter)[0].length;\n }\n return 0;\n },\n normalize(match) {\n match.url = 'https://twitter.com/' + match.url.replace(/^@/, '');\n }\n});\n\nclass IconvDecoder extends Transform {\n constructor(Iconv, charset) {\n super();\n\n // Iconv throws error on ks_c_5601-1987 when it is mapped to EUC-KR\n // https://github.com/bnoordhuis/node-iconv/issues/169\n if (charset.toLowerCase() === 'ks_c_5601-1987') {\n charset = 'CP949';\n }\n this.stream = new Iconv(charset, 'UTF-8//TRANSLIT//IGNORE');\n\n this.inputEnded = false;\n this.endCb = false;\n\n this.stream.on('error', err => this.emit('error', err));\n this.stream.on('data', chunk => this.push(chunk));\n this.stream.on('end', () => {\n this.inputEnded = true;\n if (typeof this.endCb === 'function') {\n this.endCb();\n }\n });\n }\n\n _transform(chunk, encoding, done) {\n this.stream.write(chunk);\n done();\n }\n\n _flush(done) {\n this.endCb = done;\n this.stream.end();\n }\n}\n\nclass JPDecoder extends Transform {\n constructor(charset) {\n super();\n\n this.charset = charset;\n this.chunks = [];\n this.chunklen = 0;\n }\n\n _transform(chunk, encoding, done) {\n if (typeof chunk === 'string') {\n chunk = Buffer.from(chunk, encoding);\n }\n\n this.chunks.push(chunk);\n this.chunklen += chunk.length;\n done();\n }\n\n _flush(done) {\n let input = Buffer.concat(this.chunks, this.chunklen);\n try {\n let output = encodingJapanese.convert(input, {\n to: 'UNICODE', // to_encoding\n from: this.charset, // from_encoding\n type: 'string'\n });\n if (typeof output === 'string') {\n output = Buffer.from(output);\n }\n this.push(output);\n } catch (err) {\n // keep as is on errors\n this.push(input);\n }\n\n done();\n }\n}\n\nclass MailParser extends Transform {\n constructor(config) {\n super({\n readableObjectMode: true,\n writableObjectMode: false\n });\n\n this.options = config || {};\n this.splitter = new Splitter(config);\n this.finished = false;\n this.waitingEnd = false;\n\n this.headers = false;\n this.headerLines = false;\n\n this.endReceived = false;\n this.reading = false;\n this.hasFailed = false;\n\n this.tree = false;\n this.curnode = false;\n this.waitUntilAttachmentEnd = false;\n this.attachmentCallback = false;\n\n this.hasHtml = false;\n this.hasText = false;\n\n this.text = false;\n this.html = false;\n this.textAsHtml = false;\n\n this.attachmentList = [];\n\n this.boundaries = [];\n\n this.textTypes = ['text/plain', 'text/html'].concat(!this.options.keepDeliveryStatus ? 'message/delivery-status' : []);\n\n this.decoder = this.getDecoder();\n\n this.splitter.on('readable', () => {\n if (this.reading) {\n return false;\n }\n this.readData();\n });\n\n this.splitter.on('end', () => {\n this.endReceived = true;\n if (!this.reading) {\n this.endStream();\n }\n });\n\n this.splitter.on('error', err => {\n this.hasFailed = true;\n if (typeof this.waitingEnd === 'function') {\n return this.waitingEnd(err);\n }\n this.emit('error', err);\n });\n\n this.libmime = new libmime.Libmime({ Iconv: this.options.Iconv });\n }\n\n getDecoder() {\n if (this.options.Iconv) {\n const Iconv = this.options.Iconv;\n // create wrapper\n return {\n decodeStream(charset) {\n return new IconvDecoder(Iconv, charset);\n }\n };\n } else {\n return {\n decodeStream(charset) {\n charset = (charset || 'ascii').toString().trim().toLowerCase();\n if (/^jis|^iso-?2022-?jp|^EUCJP/i.test(charset)) {\n // special case not supported by iconv-lite\n return new JPDecoder(charset);\n }\n\n return iconv.decodeStream(charset);\n }\n };\n }\n }\n\n readData() {\n if (this.hasFailed) {\n return false;\n }\n this.reading = true;\n let data = this.splitter.read();\n if (data === null) {\n this.reading = false;\n if (this.endReceived) {\n this.endStream();\n }\n return;\n }\n\n this.processChunk(data, err => {\n if (err) {\n if (typeof this.waitingEnd === 'function') {\n return this.waitingEnd(err);\n }\n return this.emit('error', err);\n }\n setImmediate(() => this.readData());\n });\n }\n\n endStream() {\n this.finished = true;\n\n if (this.curnode && this.curnode.decoder) {\n this.curnode.decoder.end();\n }\n if (typeof this.waitingEnd === 'function') {\n this.waitingEnd();\n }\n }\n\n _transform(chunk, encoding, done) {\n if (!chunk || !chunk.length) {\n return done();\n }\n\n if (this.splitter.write(chunk) === false) {\n return this.splitter.once('drain', () => {\n done();\n });\n } else {\n return done();\n }\n }\n\n _flush(done) {\n setImmediate(() => this.splitter.end());\n if (this.finished) {\n return this.cleanup(done);\n }\n this.waitingEnd = () => {\n this.cleanup(() => {\n done();\n });\n };\n }\n\n cleanup(done) {\n let finish = () => {\n try {\n let t = this.getTextContent();\n this.push(t);\n } catch (err) {\n return this.emit('error', err);\n }\n\n done();\n };\n\n if (this.curnode && this.curnode.decoder && this.curnode.decoder.readable && !this.decoderEnded) {\n (this.curnode.contentStream || this.curnode.decoder).once('end', () => {\n finish();\n });\n this.curnode.decoder.end();\n } else {\n setImmediate(() => {\n finish();\n });\n }\n }\n\n processHeaders(lines) {\n let headers = new Map();\n (lines || []).forEach(line => {\n let key = line.key;\n let value = ((this.libmime.decodeHeader(line.line) || {}).value || '').toString().trim();\n value = Buffer.from(value, 'binary').toString();\n switch (key) {\n case 'content-type':\n case 'content-disposition':\n case 'dkim-signature':\n value = this.libmime.parseHeaderValue(value);\n if (value.value) {\n value.value = this.libmime.decodeWords(value.value);\n }\n Object.keys((value && value.params) || {}).forEach(key => {\n try {\n value.params[key] = this.libmime.decodeWords(value.params[key]);\n } catch (E) {\n // ignore, keep as is\n }\n });\n break;\n case 'date': {\n let dateValue = new Date(value);\n if (isNaN(dateValue)) {\n // date parsing failed :S\n dateValue = new Date();\n }\n value = dateValue;\n break;\n }\n case 'subject':\n try {\n value = this.libmime.decodeWords(value);\n } catch (E) {\n // ignore, keep as is\n }\n break;\n case 'references':\n try {\n value = this.libmime.decodeWords(value);\n } catch (E) {\n // ignore\n }\n value = value.split(/\\s+/).map(this.ensureMessageIDFormat);\n break;\n case 'message-id':\n case 'in-reply-to':\n try {\n value = this.libmime.decodeWords(value);\n } catch (E) {\n // ignore\n }\n value = this.ensureMessageIDFormat(value);\n break;\n case 'priority':\n case 'x-priority':\n case 'x-msmail-priority':\n case 'importance':\n key = 'priority';\n value = this.parsePriority(value);\n break;\n case 'from':\n case 'to':\n case 'cc':\n case 'bcc':\n case 'sender':\n case 'reply-to':\n case 'delivered-to':\n case 'return-path':\n case 'disposition-notification-to':\n value = addressparser(value);\n this.decodeAddresses(value);\n value = {\n value,\n html: this.getAddressesHTML(value),\n text: this.getAddressesText(value)\n };\n break;\n }\n\n // handle list-* keys\n if (key.substr(0, 5) === 'list-') {\n value = this.parseListHeader(key.substr(5), value);\n key = 'list';\n }\n\n if (value) {\n if (!headers.has(key)) {\n headers.set(key, [].concat(value || []));\n } else if (Array.isArray(value)) {\n headers.set(key, headers.get(key).concat(value));\n } else {\n headers.get(key).push(value);\n }\n }\n });\n\n // keep only the first value\n let singleKeys = [\n 'message-id',\n 'content-id',\n 'from',\n 'sender',\n 'in-reply-to',\n 'reply-to',\n 'subject',\n 'date',\n 'content-disposition',\n 'content-type',\n 'content-transfer-encoding',\n 'priority',\n 'mime-version',\n 'content-description',\n 'precedence',\n 'errors-to',\n 'disposition-notification-to'\n ];\n\n headers.forEach((value, key) => {\n if (Array.isArray(value)) {\n if (singleKeys.includes(key) && value.length) {\n headers.set(key, value[value.length - 1]);\n } else if (value.length === 1) {\n headers.set(key, value[0]);\n }\n }\n\n if (key === 'list') {\n // normalize List-* headers\n let listValue = {};\n [].concat(value || []).forEach(val => {\n Object.keys(val || {}).forEach(listKey => {\n listValue[listKey] = val[listKey];\n });\n });\n headers.set(key, listValue);\n }\n });\n\n return headers;\n }\n\n parseListHeader(key, value) {\n let addresses = addressparser(value);\n let response = {};\n let data = addresses\n .map(address => {\n if (/^https?:/i.test(address.name)) {\n response.url = address.name;\n } else if (address.name) {\n response.name = address.name;\n }\n if (/^mailto:/.test(address.address)) {\n response.mail = address.address.substr(7);\n } else if (address.address && address.address.indexOf('@') < 0) {\n response.id = address.address;\n } else if (address.address) {\n response.mail = address.address;\n }\n if (Object.keys(response).length) {\n return response;\n }\n return false;\n })\n .filter(address => address);\n if (data.length) {\n return {\n [key]: response\n };\n }\n return false;\n }\n\n parsePriority(value) {\n value = value.toLowerCase().trim();\n if (!isNaN(parseInt(value, 10))) {\n // support \"X-Priority: 1 (Highest)\"\n value = parseInt(value, 10) || 0;\n if (value === 3) {\n return 'normal';\n } else if (value > 3) {\n return 'low';\n } else {\n return 'high';\n }\n } else {\n switch (value) {\n case 'non-urgent':\n case 'low':\n return 'low';\n case 'urgent':\n case 'high':\n return 'high';\n }\n }\n return 'normal';\n }\n\n ensureMessageIDFormat(value) {\n if (!value.length) {\n return false;\n }\n\n if (value.charAt(0) !== '<') {\n value = '<' + value;\n }\n\n if (value.charAt(value.length - 1) !== '>') {\n value += '>';\n }\n\n return value;\n }\n\n decodeAddresses(addresses) {\n let processedAddress = new WeakSet();\n for (let i = 0; i < addresses.length; i++) {\n let address = addresses[i];\n address.name = (address.name || '').toString().trim();\n\n if (!address.address && /^(=\\?([^?]+)\\?[Bb]\\?[^?]*\\?=)(\\s*=\\?([^?]+)\\?[Bb]\\?[^?]*\\?=)*$/.test(address.name) && !processedAddress.has(address)) {\n let parsed = addressparser(this.libmime.decodeWords(address.name));\n if (parsed.length) {\n parsed.forEach(entry => {\n processedAddress.add(entry);\n addresses.push(entry);\n });\n }\n\n // remove current element\n addresses.splice(i, 1);\n i--;\n continue;\n }\n\n if (address.name) {\n try {\n address.name = this.libmime.decodeWords(address.name);\n } catch (E) {\n //ignore, keep as is\n }\n }\n if (/@xn--/.test(address.address)) {\n try {\n address.address =\n address.address.substr(0, address.address.lastIndexOf('@') + 1) +\n punycode.toUnicode(address.address.substr(address.address.lastIndexOf('@') + 1));\n } catch (E) {\n // Not a valid punycode string; keep as is\n }\n }\n if (address.group) {\n this.decodeAddresses(address.group);\n }\n }\n }\n\n createNode(node) {\n let contentType = node.contentType;\n let disposition = node.disposition;\n let encoding = node.encoding;\n let charset = node.charset;\n\n if (!contentType && node.root) {\n contentType = 'text/plain';\n }\n\n let newNode = {\n node,\n headerLines: node.headers.lines,\n headers: this.processHeaders(node.headers.getList()),\n contentType,\n children: []\n };\n\n if (!/^multipart\\//i.test(contentType)) {\n if (disposition && !['attachment', 'inline'].includes(disposition)) {\n disposition = 'attachment';\n }\n\n if (!disposition && !this.textTypes.includes(contentType)) {\n newNode.disposition = 'attachment';\n } else {\n newNode.disposition = disposition || 'inline';\n }\n\n newNode.isAttachment = !this.textTypes.includes(contentType) || newNode.disposition !== 'inline';\n\n newNode.encoding = ['quoted-printable', 'base64'].includes(encoding) ? encoding : 'binary';\n\n if (charset) {\n newNode.charset = charset;\n }\n\n let decoder = node.getDecoder();\n decoder.on('end', () => {\n this.decoderEnded = true;\n });\n newNode.decoder = decoder;\n }\n\n if (node.root) {\n this.headers = newNode.headers;\n this.headerLines = newNode.headerLines;\n }\n\n // find location in tree\n\n if (!this.tree) {\n newNode.root = true;\n this.curnode = this.tree = newNode;\n return newNode;\n }\n\n // immediate child of root node\n if (!this.curnode.parent) {\n newNode.parent = this.curnode;\n this.curnode.children.push(newNode);\n this.curnode = newNode;\n return newNode;\n }\n\n // siblings\n if (this.curnode.parent.node === node.parentNode) {\n newNode.parent = this.curnode.parent;\n this.curnode.parent.children.push(newNode);\n this.curnode = newNode;\n return newNode;\n }\n\n // first child\n if (this.curnode.node === node.parentNode) {\n newNode.parent = this.curnode;\n this.curnode.children.push(newNode);\n this.curnode = newNode;\n return newNode;\n }\n\n // move up\n let parentNode = this.curnode;\n while ((parentNode = parentNode.parent)) {\n if (parentNode.node === node.parentNode) {\n newNode.parent = parentNode;\n parentNode.children.push(newNode);\n this.curnode = newNode;\n return newNode;\n }\n }\n\n // should never happen, can't detect parent\n this.curnode = newNode;\n return newNode;\n }\n\n getTextContent() {\n let text = [];\n let html = [];\n let processNode = (alternative, level, node) => {\n if (node.showMeta) {\n let meta = ['From', 'Subject', 'Date', 'To', 'Cc', 'Bcc']\n .map(fkey => {\n let key = fkey.toLowerCase();\n if (!node.headers.has(key)) {\n return false;\n }\n let value = node.headers.get(key);\n if (!value) {\n return false;\n }\n return {\n key: fkey,\n value: Array.isArray(value) ? value[value.length - 1] : value\n };\n })\n .filter(entry => entry);\n if (this.hasHtml) {\n html.push(\n '<table class=\"mp_head\">' +\n meta\n .map(entry => {\n let value = entry.value;\n switch (entry.key) {\n case 'From':\n case 'To':\n case 'Cc':\n case 'Bcc':\n value = value.html;\n break;\n case 'Date':\n value = this.options.formatDateString ? this.options.formatDateString(value) : value.toUTCString();\n break;\n case 'Subject':\n value = '<strong>' + he.encode(value) + '</strong>';\n break;\n default:\n value = he.encode(value);\n }\n\n return '<tr><td class=\"mp_head_key\">' + he.encode(entry.key) + ':</td><td class=\"mp_head_value\">' + value + '<td></tr>';\n })\n .join('\\n') +\n '<table>'\n );\n }\n if (this.hasText) {\n text.push(\n '\\n' +\n meta\n .map(entry => {\n let value = entry.value;\n switch (entry.key) {\n case 'From':\n case 'To':\n case 'Cc':\n case 'Bcc':\n value = value.text;\n break;\n case 'Date':\n value = this.options.formatDateString ? this.options.formatDateString(value) : value.toUTCString();\n break;\n }\n return entry.key + ': ' + value;\n })\n .join('\\n') +\n '\\n'\n );\n }\n }\n if (node.textContent) {\n if (node.contentType === 'text/plain') {\n text.push(node.textContent);\n if (!alternative && this.hasHtml) {\n html.push(this.textToHtml(node.textContent));\n }\n } else if (node.contentType === 'message/delivery-status' && !this.options.keepDeliveryStatus) {\n text.push(node.textContent);\n if (!alternative && this.hasHtml) {\n html.push(this.textToHtml(node.textContent));\n }\n } else if (node.contentType === 'text/html') {\n let failedToParseHtml = false;\n if ((!alternative && this.hasText) || (node.root && !this.hasText)) {\n if (this.options.skipHtmlToText) {\n text.push('');\n } else if (node.textContent.length > this.options.maxHtmlLengthToParse) {\n this.emit('error', new Error(`HTML too long for parsing ${node.textContent.length} bytes`));\n text.push('Invalid HTML content (too long)');\n failedToParseHtml = true;\n } else {\n try {\n text.push(htmlToText(node.textContent));\n } catch (err) {\n this.emit('error', new Error('Failed to parse HTML'));\n text.push('Invalid HTML content');\n failedToParseHtml = true;\n }\n }\n }\n if (!failedToParseHtml) {\n html.push(node.textContent);\n }\n }\n }\n alternative = alternative || node.contentType === 'multipart/alternative';\n if (node.children) {\n node.children.forEach(subNode => {\n processNode(alternative, level + 1, subNode);\n });\n }\n };\n\n processNode(false, 0, this.tree);\n\n let response = {\n type: 'text'\n };\n if (html.length) {\n this.html = response.html = html.join('<br/>\\n');\n }\n if (text.length) {\n this.text = response.text = text.join('\\n');\n this.textAsHtml = response.textAsHtml = text.map(part => this.textToHtml(part)).join('<br/>\\n');\n }\n return response;\n }\n\n processChunk(data, done) {\n let partId = null;\n if (data._parentBoundary) {\n partId = this._getPartId(data._parentBoundary);\n }\n switch (data.type) {\n case 'node': {\n let node = this.createNode(data);\n if (node === this.tree) {\n ['subject', 'references', 'date', 'to', 'from', 'to', 'cc', 'bcc', 'message-id', 'in-reply-to', 'reply-to'].forEach(key => {\n if (node.headers.has(key)) {\n this[key.replace(/-([a-z])/g, (m, c) => c.toUpperCase())] = node.headers.get(key);\n }\n });\n this.emit('headers', node.headers);\n\n if (node.headerLines) {\n this.emit('headerLines', node.headerLines);\n } \n }\n\n if (data.contentType === 'message/rfc822' && data.messageNode) {\n break;\n }\n\n if (data.parentNode && data.parentNode.contentType === 'message/rfc822') {\n node.showMeta = true;\n }\n\n if (node.isAttachment) {\n let contentType = node.contentType;\n if (node.contentType === 'application/octet-stream' && data.filename) {\n contentType = this.libmime.detectMimeType(data.filename) || 'application/octet-stream';\n }\n\n let attachment = {\n type: 'attachment',\n content: null,\n contentType,\n partId,\n release: () => {\n attachment.release = null;\n if (this.waitUntilAttachmentEnd && typeof this.attachmentCallback === 'function') {\n setImmediate(this.attachmentCallback);\n }\n this.attachmentCallback = false;\n this.waitUntilAttachmentEnd = false;\n }\n };\n\n let algo = this.options.checksumAlgo || 'md5';\n let hasher = new StreamHash(attachment, algo);\n node.decoder.on('error', err => {\n hasher.emit('error', err);\n });\n\n node.decoder.on('readable', () => {\n let chunk;\n\n while ((chunk = node.decoder.read()) !== null) {\n hasher.write(chunk);\n }\n });\n\n node.decoder.once('end', () => {\n hasher.end();\n });\n\n //node.decoder.pipe(hasher);\n attachment.content = hasher;\n\n this.waitUntilAttachmentEnd = true;\n if (data.disposition) {\n attachment.contentDisposition = data.disposition;\n }\n\n if (data.filename) {\n attachment.filename = data.filename;\n }\n\n if (node.headers.has('content-id')) {\n attachment.contentId = [].concat(node.headers.get('content-id') || []).shift();\n attachment.cid = attachment.contentId.trim().replace(/^<|>$/g, '').trim();\n // check if the attachment is \"related\" to text content like an embedded image etc\n let parentNode = node;\n while ((parentNode = parentNode.parent)) {\n if (parentNode.contentType === 'multipart/related') {\n attachment.related = true;\n }\n }\n }\n\n attachment.headers = node.headers;\n this.push(attachment);\n this.attachmentList.push(attachment);\n } else if (node.disposition === 'inline') {\n let chunks = [];\n let chunklen = 0;\n node.contentStream = node.decoder;\n\n if (node.contentType === 'text/plain') {\n this.hasText = true;\n } else if (node.contentType === 'text/html') {\n this.hasHtml = true;\n } else if (node.contentType === 'message/delivery-status' && !this.options.keepDeliveryStatus) {\n this.hasText = true;\n }\n\n if (node.node.flowed) {\n let contentStream = node.contentStream;\n let flowDecoder = new FlowedDecoder({\n delSp: node.node.delSp\n });\n contentStream.on('error', err => {\n flowDecoder.emit('error', err);\n });\n contentStream.pipe(flowDecoder);\n node.contentStream = flowDecoder;\n }\n\n let charset = node.charset || 'utf-8';\n //charset = charset || 'windows-1257';\n\n if (!['ascii', 'usascii', 'utf8'].includes(charset.toLowerCase().replace(/[^a-z0-9]+/g, ''))) {\n try {\n let contentStream = node.contentStream;\n let decodeStream = this.decoder.decodeStream(charset);\n contentStream.on('error', err => {\n decodeStream.emit('error', err);\n });\n contentStream.pipe(decodeStream);\n node.contentStream = decodeStream;\n } catch (E) {\n // do not decode charset\n }\n }\n\n node.contentStream.on('readable', () => {\n let chunk;\n while ((chunk = node.contentStream.read()) !== null) {\n if (typeof chunk === 'string') {\n chunk = Buffer.from(chunk);\n }\n chunks.push(chunk);\n chunklen += chunk.length;\n }\n });\n\n node.contentStream.once('end', () => {\n node.textContent = Buffer.concat(chunks, chunklen).toString().replace(/\\r?\\n/g, '\\n');\n });\n\n node.contentStream.once('error', err => {\n this.emit('error', err);\n });\n }\n\n break;\n }\n\n case 'data':\n if (this.curnode && this.curnode.decoder) {\n this.curnode.decoder.end();\n }\n\n if (this.waitUntilAttachmentEnd) {\n this.attachmentCallback = done;\n return;\n }\n\n // multipart message structure\n // this is not related to any specific 'node' block as it includes\n // everything between the end of some node body and between the next header\n //process.stdout.write(data.value);\n break;\n\n case 'body':\n if (this.curnode && this.curnode.decoder && this.curnode.decoder.writable) {\n if (this.curnode.decoder.write(data.value) === false) {\n return this.curnode.decoder.once('drain', done);\n }\n }\n\n // Leaf element body. Includes the body for the last 'node' block. You might\n // have several 'body' calls for a single 'node' block\n //process.stdout.write(data.value);\n break;\n }\n\n setImmediate(done);\n }\n\n _getPartId(parentBoundary) {\n let boundaryIndex = this.boundaries.findIndex(item => item.name === parentBoundary);\n if (boundaryIndex === -1) {\n this.boundaries.push({ name: parentBoundary, count: 1 });\n boundaryIndex = this.boundaries.length - 1;\n } else {\n this.boundaries[boundaryIndex].count++;\n }\n let partId = '1';\n for (let i = 0; i <= boundaryIndex; i++) {\n if (i === 0) partId = this.boundaries[i].count.toString();\n else partId += '.' + this.boundaries[i].count.toString();\n }\n return partId;\n }\n\n getAddressesHTML(value) {\n let formatSingleLevel = addresses =>\n addresses\n .map(address => {\n let str = '<span class=\"mp_address_group\">';\n if (address.name) {\n str += '<span class=\"mp_address_name\">' + he.encode(address.name) + (address.group ? ': ' : '') + '</span>';\n }\n if (address.address) {\n let link = '<a href=\"mailto:' + he.encode(address.address) + '\" class=\"mp_address_email\">' + he.encode(address.address) + '</a>';\n if (address.name) {\n str += ' <' + link + '>';\n } else {\n str += link;\n }\n }\n if (address.group) {\n str += formatSingleLevel(address.group) + ';';\n }\n return str + '</span>';\n })\n .join(', ');\n return formatSingleLevel([].concat(value || []));\n }\n\n getAddressesText(value) {\n let formatSingleLevel = addresses =>\n addresses\n .map(address => {\n let str = '';\n if (address.name) {\n str += `\"${address.name}\"` + (address.group ? ': ' : '');\n }\n if (address.address) {\n let link = address.address;\n if (address.name) {\n str += ' <' + link + '>';\n } else {\n str += link;\n }\n }\n if (address.group) {\n str += formatSingleLevel(address.group) + ';';\n }\n return str;\n })\n .join(', ');\n return formatSingleLevel([].concat(value || []));\n }\n\n updateImageLinks(replaceCallback, done) {\n if (!this.html) {\n return setImmediate(() => done(null, false));\n }\n\n let cids = new Map();\n let html = (this.html || '').toString();\n\n if (this.options.skipImageLinks) {\n return done(null, html);\n }\n\n html.replace(/\\bcid:([^'\"\\s]{1,256})/g, (match, cid) => {\n for (let i = 0, len = this.attachmentList.length; i < len; i++) {\n if (this.attachmentList[i].cid === cid && /^image\\/[\\w]+$/i.test(this.attachmentList[i].contentType)) {\n cids.set(cid, {\n attachment: this.attachmentList[i]\n });\n break;\n }\n }\n return match;\n });\n\n let cidList = [];\n cids.forEach(entry => {\n cidList.push(entry);\n });\n\n let pos = 0;\n let processNext = () => {\n if (pos >= cidList.length) {\n html = html.replace(/\\bcid:([^'\"\\s]{1,256})/g, (match, cid) => {\n if (cids.has(cid) && cids.get(cid).url) {\n return cids.get(cid).url;\n }\n return match;\n });\n\n return done(null, html);\n }\n let entry = cidList[pos++];\n replaceCallback(entry.attachment, (err, url) => {\n if (err) {\n return setImmediate(() => done(err));\n }\n entry.url = url;\n setImmediate(processNext);\n });\n };\n\n setImmediate(processNext);\n }\n\n textToHtml(str) {\n if (this.options.skipTextToHtml) {\n return '';\n }\n str = (str || '').toString();\n let encoded;\n\n let linkified = false;\n if (!this.options.skipTextLinks) {\n try {\n if (linkify.pretest(str)) {\n linkified = true;\n let links = linkify.match(str) || [];\n let result = [];\n let last = 0;\n\n links.forEach(link => {\n if (last < link.index) {\n let textPart = he\n // encode special chars\n .encode(str.slice(last, link.index), {\n useNamedReferences: true\n });\n result.push(textPart);\n }\n\n result.push(`<a href=\"${link.url}\">${link.text}</a>`);\n\n last = link.lastIndex;\n });\n\n let textPart = he\n // encode special chars\n .encode(str.slice(last), {\n useNamedReferences: true\n });\n result.push(textPart);\n\n encoded = result.join('');\n }\n } catch (E) {\n // failed, don't linkify\n }\n }\n\n if (!linkified) {\n encoded = he\n // encode special chars\n .encode(str, {\n useNamedReferences: true\n });\n }\n\n let text =\n '<p>' +\n encoded\n .replace(/\\r?\\n/g, '\\n')\n .trim() // normalize line endings\n .replace(/[ \\t]+$/gm, '')\n .trim() // trim empty line endings\n .replace(/\\n\\n+/g, '</p><p>')\n .trim() // insert <p> to multiple linebreaks\n .replace(/\\n/g, '<br/>') + // insert <br> to single linebreaks\n '</p>';\n\n return text;\n }\n}\n\nmodule.exports = MailParser;\n//# sourceURL=[module]\n//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"file":"(rsc)/./node_modules/mailparser/lib/mail-parser.js","mappings":"AAAa;;AAEb,kBAAkB,mBAAO,CAAC,0DAAW;AACrC,gBAAgB,mBAAO,CAAC,4DAAS;AACjC,sBAAsB,mBAAO,CAAC,wHAA8B;AAC5D,kBAAkB,uDAA2B;AAC7C;AACA,iBAAiB,mBAAO,CAAC,qEAAa;AACtC,sBAAsB,mBAAO,CAAC,0FAA8B;AAC5D,mBAAmB,mBAAO,CAAC,yEAAe;AAC1C,cAAc,mBAAO,CAAC,gEAAY;AAClC,QAAQ,aAAa,EAAE,mBAAO,CAAC,4EAAc;AAC7C,WAAW,mBAAO,CAAC,yCAAI;AACvB,gBAAgB,mBAAO,CAAC,sEAAY;AACpC,aAAa,mBAAO,CAAC,kDAAM;AAC3B,yBAAyB,mBAAO,CAAC,8EAAmB;;AAEpD;AACA;AACA;AACA;AACA;AACA,WAAW,kDAAkD;;AAE7D;AACA;AACA;AACA;AACA;;AAEA;AACA,0DAA0D,KAAK;AAC/D;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,KAAK;AACL;AACA;AACA;AACA,CAAC;;AAED;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,SAAS;AACT;;AAEA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;;AAEA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,aAAa;AACb;AACA;AACA;AACA;AACA,UAAU;AACV;AACA;AACA;;AAEA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA,SAAS;;AAET;AACA;AACA;AACA;;AAEA;AACA;;AAEA;AACA;AACA;;AAEA;AACA;AACA;AACA;;AAEA;AACA;;AAEA;AACA;AACA;;AAEA;;AAEA;;AAEA;;AAEA;;AAEA;AACA;AACA;AACA;AACA;AACA,SAAS;;AAET;AACA;AACA;AACA;AACA;AACA,SAAS;;AAET;AACA;AACA;AACA;AACA;AACA;AACA,SAAS;;AAET,6CAA6C,2BAA2B;AACxE;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,UAAU;AACV;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,SAAS;AACT;;AAEA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA,aAAa;AACb,UAAU;AACV;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,aAAa;AACb;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA,cAAc;AACd;AACA;;AAEA;AACA;;AAEA;AACA;AACA;AACA,aAAa;AACb;AACA,UAAU;AACV;AACA;AACA,aAAa;AACb;AACA;;AAEA;AACA;AACA;AACA;AACA,oEAAoE;AACpE;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,6DAA6D;AAC7D;AACA;AACA,0BAA0B;AAC1B;AACA;AACA,qBAAqB;AACrB;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,sBAAsB;AACtB;AACA;AACA;AACA;AACA;AACA;AACA,sBAAsB;AACtB;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,sBAAsB;AACtB;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA,kBAAkB;AAClB;AACA,kBAAkB;AAClB;AACA;AACA;AACA,SAAS;;AAET;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA,kBAAkB;AAClB;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA,yCAAyC;AACzC;AACA,qBAAqB;AACrB,iBAAiB;AACjB;AACA;AACA,SAAS;;AAET;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,kBAAkB;AAClB;AACA;AACA;AACA;AACA,kBAAkB;AAClB;AACA,kBAAkB;AAClB;AACA;AACA;AACA;AACA;AACA;AACA,aAAa;AACb;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,cAAc;AACd;AACA,cAAc;AACd;AACA;AACA,UAAU;AACV;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;;AAEA;AACA;AACA;;AAEA;AACA;AACA;;AAEA;AACA;;AAEA;AACA;AACA,wBAAwB,sBAAsB;AAC9C;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA,qBAAqB;AACrB;;AAEA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA,kBAAkB;AAClB;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,kBAAkB;AAClB,oDAAoD;AACpD;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;;AAEA;AACA;AACA,cAAc;AACd;AACA;;AAEA;;AAEA;;AAEA;AACA;AACA;;AAEA;AACA;AACA;AACA,aAAa;AACb;AACA;;AAEA;AACA;AACA;AACA;;AAEA;;AAEA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,qBAAqB;AACrB;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA,iCAAiC;AACjC;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,iCAAiC;AACjC;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,kBAAkB;AAClB;AACA;AACA;AACA;AACA,kBAAkB;AAClB;AACA;AACA;AACA;AACA,0BAA0B;AAC1B,sFAAsF,yBAAyB;AAC/G;AACA;AACA,0BAA0B;AAC1B;AACA;AACA,8BAA8B;AAC9B;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,iBAAiB;AACjB;AACA;;AAEA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,qBAAqB;AACrB;;AAEA;AACA;AACA;AACA;;AAEA;AACA;AACA;;AAEA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA,qBAAqB;;AAErB;AACA;;AAEA;AACA;AACA;AACA,qBAAqB;;AAErB;AACA;AACA,qBAAqB;;AAErB;AACA;;AAEA;AACA;AACA;AACA;;AAEA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA,kBAAkB;AAClB;AACA;AACA;;AAEA;AACA;AACA,sBAAsB;AACtB;AACA,sBAAsB;AACtB;AACA;;AAEA;AACA;AACA;AACA;AACA,yBAAyB;AACzB;AACA;AACA,yBAAyB;AACzB;AACA;AACA;;AAEA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA,6BAA6B;AAC7B;AACA;AACA,0BAA0B;AAC1B;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,qBAAqB;;AAErB;AACA;AACA,qBAAqB;;AAErB;AACA;AACA,qBAAqB;AACrB;;AAEA;AACA;;AAEA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;;AAEA;AACA;;AAEA;AACA;AACA;AACA,mCAAmC,gCAAgC;AACnE;AACA,UAAU;AACV;AACA;AACA;AACA,wBAAwB,oBAAoB;AAC5C;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,yCAAyC,gBAAgB;AACzD,0BAA0B;AAC1B;AACA;AACA;AACA;AACA,oEAAoE;AACpE;AACA;AACA,iBAAiB;AACjB;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA,mCAAmC,aAAa;AAChD;AACA;AACA;AACA;AACA;AACA,0BAA0B;AAC1B;AACA;AACA;AACA;AACA,oEAAoE;AACpE;AACA;AACA,iBAAiB;AACjB;AACA;AACA;;AAEA;AACA;AACA;AACA;;AAEA;AACA;;AAEA;AACA;AACA;;AAEA,qCAAqC,MAAM;AAC3C,8DAA8D,SAAS;AACvE;AACA;AACA;AACA,qBAAqB;AACrB;AACA;AACA;AACA;AACA,SAAS;;AAET;AACA;AACA;AACA,SAAS;;AAET;AACA;AACA;AACA,oDAAoD,MAAM;AAC1D;AACA;AACA;AACA;AACA,iBAAiB;;AAEjB;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,aAAa;AACb;;AAEA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA,iCAAiC;AACjC;AACA;;AAEA,gDAAgD,SAAS,IAAI,UAAU;;AAEvE;AACA,qBAAqB;;AAErB;AACA;AACA;AACA;AACA,yBAAyB;AACzB;;AAEA;AACA;AACA,cAAc;AACd;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA,iBAAiB;AACjB;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;;AAEA","sources":["/home/alma/nextgen/Neah-mail/node_modules/mailparser/lib/mail-parser.js"],"sourcesContent":["'use strict';\n\nconst mailsplit = require('mailsplit');\nconst libmime = require('libmime');\nconst addressparser = require('nodemailer/lib/addressparser');\nconst Transform = require('stream').Transform;\nconst Splitter = mailsplit.Splitter;\nconst punycode = require('punycode.js');\nconst FlowedDecoder = require('mailsplit/lib/flowed-decoder');\nconst StreamHash = require('./stream-hash');\nconst iconv = require('iconv-lite');\nconst { htmlToText } = require('html-to-text');\nconst he = require('he');\nconst linkify = require('linkify-it')();\nconst tlds = require('tlds');\nconst encodingJapanese = require('encoding-japanese');\n\nlinkify\n    .tlds(tlds) // Reload with full tlds list\n    .tlds('onion', true) // Add unofficial `.onion` domain\n    .add('git:', 'http:') // Add `git:` ptotocol as \"alias\"\n    .add('ftp:', null) // Disable `ftp:` ptotocol\n    .set({ fuzzyIP: true, fuzzyLink: true, fuzzyEmail: true });\n\n// twitter linkifier from\n// https://github.com/markdown-it/linkify-it#example-2-add-twitter-mentions-handler\nlinkify.add('@', {\n    validate(text, pos, self) {\n        let tail = text.slice(pos);\n\n        if (!self.re.twitter) {\n            self.re.twitter = new RegExp('^([a-zA-Z0-9_]){1,15}(?!_)(?=$|' + self.re.src_ZPCc + ')');\n        }\n        if (self.re.twitter.test(tail)) {\n            // Linkifier allows punctuation chars before prefix,\n            // but we additionally disable `@` (\"@@mention\" is invalid)\n            if (pos >= 2 && tail[pos - 2] === '@') {\n                return false;\n            }\n            return tail.match(self.re.twitter)[0].length;\n        }\n        return 0;\n    },\n    normalize(match) {\n        match.url = 'https://twitter.com/' + match.url.replace(/^@/, '');\n    }\n});\n\nclass IconvDecoder extends Transform {\n    constructor(Iconv, charset) {\n        super();\n\n        // Iconv throws error on ks_c_5601-1987 when it is mapped to EUC-KR\n        // https://github.com/bnoordhuis/node-iconv/issues/169\n        if (charset.toLowerCase() === 'ks_c_5601-1987') {\n            charset = 'CP949';\n        }\n        this.stream = new Iconv(charset, 'UTF-8//TRANSLIT//IGNORE');\n\n        this.inputEnded = false;\n        this.endCb = false;\n\n        this.stream.on('error', err => this.emit('error', err));\n        this.stream.on('data', chunk => this.push(chunk));\n        this.stream.on('end', () => {\n            this.inputEnded = true;\n            if (typeof this.endCb === 'function') {\n                this.endCb();\n            }\n        });\n    }\n\n    _transform(chunk, encoding, done) {\n        this.stream.write(chunk);\n        done();\n    }\n\n    _flush(done) {\n        this.endCb = done;\n        this.stream.end();\n    }\n}\n\nclass JPDecoder extends Transform {\n    constructor(charset) {\n        super();\n\n        this.charset = charset;\n        this.chunks = [];\n        this.chunklen = 0;\n    }\n\n    _transform(chunk, encoding, done) {\n        if (typeof chunk === 'string') {\n            chunk = Buffer.from(chunk, encoding);\n        }\n\n        this.chunks.push(chunk);\n        this.chunklen += chunk.length;\n        done();\n    }\n\n    _flush(done) {\n        let input = Buffer.concat(this.chunks, this.chunklen);\n        try {\n            let output = encodingJapanese.convert(input, {\n                to: 'UNICODE', // to_encoding\n                from: this.charset, // from_encoding\n                type: 'string'\n            });\n            if (typeof output === 'string') {\n                output = Buffer.from(output);\n            }\n            this.push(output);\n        } catch (err) {\n            // keep as is on errors\n            this.push(input);\n        }\n\n        done();\n    }\n}\n\nclass MailParser extends Transform {\n    constructor(config) {\n        super({\n            readableObjectMode: true,\n            writableObjectMode: false\n        });\n\n        this.options = config || {};\n        this.splitter = new Splitter(config);\n        this.finished = false;\n        this.waitingEnd = false;\n\n        this.headers = false;\n        this.headerLines = false;\n\n        this.endReceived = false;\n        this.reading = false;\n        this.hasFailed = false;\n\n        this.tree = false;\n        this.curnode = false;\n        this.waitUntilAttachmentEnd = false;\n        this.attachmentCallback = false;\n\n        this.hasHtml = false;\n        this.hasText = false;\n\n        this.text = false;\n        this.html = false;\n        this.textAsHtml = false;\n\n        this.attachmentList = [];\n\n        this.boundaries = [];\n\n        this.textTypes = ['text/plain', 'text/html'].concat(!this.options.keepDeliveryStatus ? 'message/delivery-status' : []);\n\n        this.decoder = this.getDecoder();\n\n        this.splitter.on('readable', () => {\n            if (this.reading) {\n                return false;\n            }\n            this.readData();\n        });\n\n        this.splitter.on('end', () => {\n            this.endReceived = true;\n            if (!this.reading) {\n                this.endStream();\n            }\n        });\n\n        this.splitter.on('error', err => {\n            this.hasFailed = true;\n            if (typeof this.waitingEnd === 'function') {\n                return this.waitingEnd(err);\n            }\n            this.emit('error', err);\n        });\n\n        this.libmime = new libmime.Libmime({ Iconv: this.options.Iconv });\n    }\n\n    getDecoder() {\n        if (this.options.Iconv) {\n            const Iconv = this.options.Iconv;\n            // create wrapper\n            return {\n                decodeStream(charset) {\n                    return new IconvDecoder(Iconv, charset);\n                }\n            };\n        } else {\n            return {\n                decodeStream(charset) {\n                    charset = (charset || 'ascii').toString().trim().toLowerCase();\n                    if (/^jis|^iso-?2022-?jp|^EUCJP/i.test(charset)) {\n                        // special case not supported by iconv-lite\n                        return new JPDecoder(charset);\n                    }\n\n                    return iconv.decodeStream(charset);\n                }\n            };\n        }\n    }\n\n    readData() {\n        if (this.hasFailed) {\n            return false;\n        }\n        this.reading = true;\n        let data = this.splitter.read();\n        if (data === null) {\n            this.reading = false;\n            if (this.endReceived) {\n                this.endStream();\n            }\n            return;\n        }\n\n        this.processChunk(data, err => {\n            if (err) {\n                if (typeof this.waitingEnd === 'function') {\n                    return this.waitingEnd(err);\n                }\n                return this.emit('error', err);\n            }\n            setImmediate(() => this.readData());\n        });\n    }\n\n    endStream() {\n        this.finished = true;\n\n        if (this.curnode && this.curnode.decoder) {\n            this.curnode.decoder.end();\n        }\n        if (typeof this.waitingEnd === 'function') {\n            this.waitingEnd();\n        }\n    }\n\n    _transform(chunk, encoding, done) {\n        if (!chunk || !chunk.length) {\n            return done();\n        }\n\n        if (this.splitter.write(chunk) === false) {\n            return this.splitter.once('drain', () => {\n                done();\n            });\n        } else {\n            return done();\n        }\n    }\n\n    _flush(done) {\n        setImmediate(() => this.splitter.end());\n        if (this.finished) {\n            return this.cleanup(done);\n        }\n        this.waitingEnd = () => {\n            this.cleanup(() => {\n                done();\n            });\n        };\n    }\n\n    cleanup(done) {\n        let finish = () => {\n            try {\n                let t = this.getTextContent();\n                this.push(t);\n            } catch (err) {\n                return this.emit('error', err);\n            }\n\n            done();\n        };\n\n        if (this.curnode && this.curnode.decoder && this.curnode.decoder.readable && !this.decoderEnded) {\n            (this.curnode.contentStream || this.curnode.decoder).once('end', () => {\n                finish();\n            });\n            this.curnode.decoder.end();\n        } else {\n            setImmediate(() => {\n                finish();\n            });\n        }\n    }\n\n    processHeaders(lines) {\n        let headers = new Map();\n        (lines || []).forEach(line => {\n            let key = line.key;\n            let value = ((this.libmime.decodeHeader(line.line) || {}).value || '').toString().trim();\n            value = Buffer.from(value, 'binary').toString();\n            switch (key) {\n                case 'content-type':\n                case 'content-disposition':\n                case 'dkim-signature':\n                    value = this.libmime.parseHeaderValue(value);\n                    if (value.value) {\n                        value.value = this.libmime.decodeWords(value.value);\n                    }\n                    Object.keys((value && value.params) || {}).forEach(key => {\n                        try {\n                            value.params[key] = this.libmime.decodeWords(value.params[key]);\n                        } catch (E) {\n                            // ignore, keep as is\n                        }\n                    });\n                    break;\n                case 'date': {\n                    let dateValue = new Date(value);\n                    if (isNaN(dateValue)) {\n                        // date parsing failed :S\n                        dateValue = new Date();\n                    }\n                    value = dateValue;\n                    break;\n                }\n                case 'subject':\n                    try {\n                        value = this.libmime.decodeWords(value);\n                    } catch (E) {\n                        // ignore, keep as is\n                    }\n                    break;\n                case 'references':\n                    try {\n                        value = this.libmime.decodeWords(value);\n                    } catch (E) {\n                        // ignore\n                    }\n                    value = value.split(/\\s+/).map(this.ensureMessageIDFormat);\n                    break;\n                case 'message-id':\n                case 'in-reply-to':\n                    try {\n                        value = this.libmime.decodeWords(value);\n                    } catch (E) {\n                        // ignore\n                    }\n                    value = this.ensureMessageIDFormat(value);\n                    break;\n                case 'priority':\n                case 'x-priority':\n                case 'x-msmail-priority':\n                case 'importance':\n                    key = 'priority';\n                    value = this.parsePriority(value);\n                    break;\n                case 'from':\n                case 'to':\n                case 'cc':\n                case 'bcc':\n                case 'sender':\n                case 'reply-to':\n                case 'delivered-to':\n                case 'return-path':\n                case 'disposition-notification-to':\n                    value = addressparser(value);\n                    this.decodeAddresses(value);\n                    value = {\n                        value,\n                        html: this.getAddressesHTML(value),\n                        text: this.getAddressesText(value)\n                    };\n                    break;\n            }\n\n            // handle list-* keys\n            if (key.substr(0, 5) === 'list-') {\n                value = this.parseListHeader(key.substr(5), value);\n                key = 'list';\n            }\n\n            if (value) {\n                if (!headers.has(key)) {\n                    headers.set(key, [].concat(value || []));\n                } else if (Array.isArray(value)) {\n                    headers.set(key, headers.get(key).concat(value));\n                } else {\n                    headers.get(key).push(value);\n                }\n            }\n        });\n\n        // keep only the first value\n        let singleKeys = [\n            'message-id',\n            'content-id',\n            'from',\n            'sender',\n            'in-reply-to',\n            'reply-to',\n            'subject',\n            'date',\n            'content-disposition',\n            'content-type',\n            'content-transfer-encoding',\n            'priority',\n            'mime-version',\n            'content-description',\n            'precedence',\n            'errors-to',\n            'disposition-notification-to'\n        ];\n\n        headers.forEach((value, key) => {\n            if (Array.isArray(value)) {\n                if (singleKeys.includes(key) && value.length) {\n                    headers.set(key, value[value.length - 1]);\n                } else if (value.length === 1) {\n                    headers.set(key, value[0]);\n                }\n            }\n\n            if (key === 'list') {\n                // normalize List-* headers\n                let listValue = {};\n                [].concat(value || []).forEach(val => {\n                    Object.keys(val || {}).forEach(listKey => {\n                        listValue[listKey] = val[listKey];\n                    });\n                });\n                headers.set(key, listValue);\n            }\n        });\n\n        return headers;\n    }\n\n    parseListHeader(key, value) {\n        let addresses = addressparser(value);\n        let response = {};\n        let data = addresses\n            .map(address => {\n                if (/^https?:/i.test(address.name)) {\n                    response.url = address.name;\n                } else if (address.name) {\n                    response.name = address.name;\n                }\n                if (/^mailto:/.test(address.address)) {\n                    response.mail = address.address.substr(7);\n                } else if (address.address && address.address.indexOf('@') < 0) {\n                    response.id = address.address;\n                } else if (address.address) {\n                    response.mail = address.address;\n                }\n                if (Object.keys(response).length) {\n                    return response;\n                }\n                return false;\n            })\n            .filter(address => address);\n        if (data.length) {\n            return {\n                [key]: response\n            };\n        }\n        return false;\n    }\n\n    parsePriority(value) {\n        value = value.toLowerCase().trim();\n        if (!isNaN(parseInt(value, 10))) {\n            // support \"X-Priority: 1 (Highest)\"\n            value = parseInt(value, 10) || 0;\n            if (value === 3) {\n                return 'normal';\n            } else if (value > 3) {\n                return 'low';\n            } else {\n                return 'high';\n            }\n        } else {\n            switch (value) {\n                case 'non-urgent':\n                case 'low':\n                    return 'low';\n                case 'urgent':\n                case 'high':\n                    return 'high';\n            }\n        }\n        return 'normal';\n    }\n\n    ensureMessageIDFormat(value) {\n        if (!value.length) {\n            return false;\n        }\n\n        if (value.charAt(0) !== '<') {\n            value = '<' + value;\n        }\n\n        if (value.charAt(value.length - 1) !== '>') {\n            value += '>';\n        }\n\n        return value;\n    }\n\n    decodeAddresses(addresses) {\n        let processedAddress = new WeakSet();\n        for (let i = 0; i < addresses.length; i++) {\n            let address = addresses[i];\n            address.name = (address.name || '').toString().trim();\n\n            if (!address.address && /^(=\\?([^?]+)\\?[Bb]\\?[^?]*\\?=)(\\s*=\\?([^?]+)\\?[Bb]\\?[^?]*\\?=)*$/.test(address.name) && !processedAddress.has(address)) {\n                let parsed = addressparser(this.libmime.decodeWords(address.name));\n                if (parsed.length) {\n                    parsed.forEach(entry => {\n                        processedAddress.add(entry);\n                        addresses.push(entry);\n                    });\n                }\n\n                // remove current element\n                addresses.splice(i, 1);\n                i--;\n                continue;\n            }\n\n            if (address.name) {\n                try {\n                    address.name = this.libmime.decodeWords(address.name);\n                } catch (E) {\n                    //ignore, keep as is\n                }\n            }\n            if (/@xn--/.test(address.address)) {\n                try {\n                    address.address =\n                        address.address.substr(0, address.address.lastIndexOf('@') + 1) +\n                        punycode.toUnicode(address.address.substr(address.address.lastIndexOf('@') + 1));\n                } catch (E) {\n                    // Not a valid punycode string; keep as is\n                }\n            }\n            if (address.group) {\n                this.decodeAddresses(address.group);\n            }\n        }\n    }\n\n    createNode(node) {\n        let contentType = node.contentType;\n        let disposition = node.disposition;\n        let encoding = node.encoding;\n        let charset = node.charset;\n\n        if (!contentType && node.root) {\n            contentType = 'text/plain';\n        }\n\n        let newNode = {\n            node,\n            headerLines: node.headers.lines,\n            headers: this.processHeaders(node.headers.getList()),\n            contentType,\n            children: []\n        };\n\n        if (!/^multipart\\//i.test(contentType)) {\n            if (disposition && !['attachment', 'inline'].includes(disposition)) {\n                disposition = 'attachment';\n            }\n\n            if (!disposition && !this.textTypes.includes(contentType)) {\n                newNode.disposition = 'attachment';\n            } else {\n                newNode.disposition = disposition || 'inline';\n            }\n\n            newNode.isAttachment = !this.textTypes.includes(contentType) || newNode.disposition !== 'inline';\n\n            newNode.encoding = ['quoted-printable', 'base64'].includes(encoding) ? encoding : 'binary';\n\n            if (charset) {\n                newNode.charset = charset;\n            }\n\n            let decoder = node.getDecoder();\n            decoder.on('end', () => {\n                this.decoderEnded = true;\n            });\n            newNode.decoder = decoder;\n        }\n\n        if (node.root) {\n            this.headers = newNode.headers;\n            this.headerLines = newNode.headerLines;\n        }\n\n        // find location in tree\n\n        if (!this.tree) {\n            newNode.root = true;\n            this.curnode = this.tree = newNode;\n            return newNode;\n        }\n\n        // immediate child of root node\n        if (!this.curnode.parent) {\n            newNode.parent = this.curnode;\n            this.curnode.children.push(newNode);\n            this.curnode = newNode;\n            return newNode;\n        }\n\n        // siblings\n        if (this.curnode.parent.node === node.parentNode) {\n            newNode.parent = this.curnode.parent;\n            this.curnode.parent.children.push(newNode);\n            this.curnode = newNode;\n            return newNode;\n        }\n\n        // first child\n        if (this.curnode.node === node.parentNode) {\n            newNode.parent = this.curnode;\n            this.curnode.children.push(newNode);\n            this.curnode = newNode;\n            return newNode;\n        }\n\n        // move up\n        let parentNode = this.curnode;\n        while ((parentNode = parentNode.parent)) {\n            if (parentNode.node === node.parentNode) {\n                newNode.parent = parentNode;\n                parentNode.children.push(newNode);\n                this.curnode = newNode;\n                return newNode;\n            }\n        }\n\n        // should never happen, can't detect parent\n        this.curnode = newNode;\n        return newNode;\n    }\n\n    getTextContent() {\n        let text = [];\n        let html = [];\n        let processNode = (alternative, level, node) => {\n            if (node.showMeta) {\n                let meta = ['From', 'Subject', 'Date', 'To', 'Cc', 'Bcc']\n                    .map(fkey => {\n                        let key = fkey.toLowerCase();\n                        if (!node.headers.has(key)) {\n                            return false;\n                        }\n                        let value = node.headers.get(key);\n                        if (!value) {\n                            return false;\n                        }\n                        return {\n                            key: fkey,\n                            value: Array.isArray(value) ? value[value.length - 1] : value\n                        };\n                    })\n                    .filter(entry => entry);\n                if (this.hasHtml) {\n                    html.push(\n                        '<table class=\"mp_head\">' +\n                            meta\n                                .map(entry => {\n                                    let value = entry.value;\n                                    switch (entry.key) {\n                                        case 'From':\n                                        case 'To':\n                                        case 'Cc':\n                                        case 'Bcc':\n                                            value = value.html;\n                                            break;\n                                        case 'Date':\n                                            value = this.options.formatDateString ? this.options.formatDateString(value) : value.toUTCString();\n                                            break;\n                                        case 'Subject':\n                                            value = '<strong>' + he.encode(value) + '</strong>';\n                                            break;\n                                        default:\n                                            value = he.encode(value);\n                                    }\n\n                                    return '<tr><td class=\"mp_head_key\">' + he.encode(entry.key) + ':</td><td class=\"mp_head_value\">' + value + '<td></tr>';\n                                })\n                                .join('\\n') +\n                            '<table>'\n                    );\n                }\n                if (this.hasText) {\n                    text.push(\n                        '\\n' +\n                            meta\n                                .map(entry => {\n                                    let value = entry.value;\n                                    switch (entry.key) {\n                                        case 'From':\n                                        case 'To':\n                                        case 'Cc':\n                                        case 'Bcc':\n                                            value = value.text;\n                                            break;\n                                        case 'Date':\n                                            value = this.options.formatDateString ? this.options.formatDateString(value) : value.toUTCString();\n                                            break;\n                                    }\n                                    return entry.key + ': ' + value;\n                                })\n                                .join('\\n') +\n                            '\\n'\n                    );\n                }\n            }\n            if (node.textContent) {\n                if (node.contentType === 'text/plain') {\n                    text.push(node.textContent);\n                    if (!alternative && this.hasHtml) {\n                        html.push(this.textToHtml(node.textContent));\n                    }\n                } else if (node.contentType === 'message/delivery-status' && !this.options.keepDeliveryStatus) {\n                    text.push(node.textContent);\n                    if (!alternative && this.hasHtml) {\n                        html.push(this.textToHtml(node.textContent));\n                    }\n                } else if (node.contentType === 'text/html') {\n                    let failedToParseHtml = false;\n                    if ((!alternative && this.hasText) || (node.root && !this.hasText)) {\n                        if (this.options.skipHtmlToText) {\n                            text.push('');\n                        } else if (node.textContent.length > this.options.maxHtmlLengthToParse) {\n                            this.emit('error', new Error(`HTML too long for parsing ${node.textContent.length} bytes`));\n                            text.push('Invalid HTML content (too long)');\n                            failedToParseHtml = true;\n                        } else {\n                            try {\n                                text.push(htmlToText(node.textContent));\n                            } catch (err) {\n                                this.emit('error', new Error('Failed to parse HTML'));\n                                text.push('Invalid HTML content');\n                                failedToParseHtml = true;\n                            }\n                        }\n                    }\n                    if (!failedToParseHtml) {\n                        html.push(node.textContent);\n                    }\n                }\n            }\n            alternative = alternative || node.contentType === 'multipart/alternative';\n            if (node.children) {\n                node.children.forEach(subNode => {\n                    processNode(alternative, level + 1, subNode);\n                });\n            }\n        };\n\n        processNode(false, 0, this.tree);\n\n        let response = {\n            type: 'text'\n        };\n        if (html.length) {\n            this.html = response.html = html.join('<br/>\\n');\n        }\n        if (text.length) {\n            this.text = response.text = text.join('\\n');\n            this.textAsHtml = response.textAsHtml = text.map(part => this.textToHtml(part)).join('<br/>\\n');\n        }\n        return response;\n    }\n\n    processChunk(data, done) {\n        let partId = null;\n        if (data._parentBoundary) {\n            partId = this._getPartId(data._parentBoundary);\n        }\n        switch (data.type) {\n            case 'node': {\n                let node = this.createNode(data);\n                if (node === this.tree) {\n                    ['subject', 'references', 'date', 'to', 'from', 'to', 'cc', 'bcc', 'message-id', 'in-reply-to', 'reply-to'].forEach(key => {\n                        if (node.headers.has(key)) {\n                            this[key.replace(/-([a-z])/g, (m, c) => c.toUpperCase())] = node.headers.get(key);\n                        }\n                    });\n                    this.emit('headers', node.headers);\n\n                    if (node.headerLines) {\n                        this.emit('headerLines', node.headerLines);\n                    }                    \n                }\n\n                if (data.contentType === 'message/rfc822' && data.messageNode) {\n                    break;\n                }\n\n                if (data.parentNode && data.parentNode.contentType === 'message/rfc822') {\n                    node.showMeta = true;\n                }\n\n                if (node.isAttachment) {\n                    let contentType = node.contentType;\n                    if (node.contentType === 'application/octet-stream' && data.filename) {\n                        contentType = this.libmime.detectMimeType(data.filename) || 'application/octet-stream';\n                    }\n\n                    let attachment = {\n                        type: 'attachment',\n                        content: null,\n                        contentType,\n                        partId,\n                        release: () => {\n                            attachment.release = null;\n                            if (this.waitUntilAttachmentEnd && typeof this.attachmentCallback === 'function') {\n                                setImmediate(this.attachmentCallback);\n                            }\n                            this.attachmentCallback = false;\n                            this.waitUntilAttachmentEnd = false;\n                        }\n                    };\n\n                    let algo = this.options.checksumAlgo || 'md5';\n                    let hasher = new StreamHash(attachment, algo);\n                    node.decoder.on('error', err => {\n                        hasher.emit('error', err);\n                    });\n\n                    node.decoder.on('readable', () => {\n                        let chunk;\n\n                        while ((chunk = node.decoder.read()) !== null) {\n                            hasher.write(chunk);\n                        }\n                    });\n\n                    node.decoder.once('end', () => {\n                        hasher.end();\n                    });\n\n                    //node.decoder.pipe(hasher);\n                    attachment.content = hasher;\n\n                    this.waitUntilAttachmentEnd = true;\n                    if (data.disposition) {\n                        attachment.contentDisposition = data.disposition;\n                    }\n\n                    if (data.filename) {\n                        attachment.filename = data.filename;\n                    }\n\n                    if (node.headers.has('content-id')) {\n                        attachment.contentId = [].concat(node.headers.get('content-id') || []).shift();\n                        attachment.cid = attachment.contentId.trim().replace(/^<|>$/g, '').trim();\n                        // check if the attachment is \"related\" to text content like an embedded image etc\n                        let parentNode = node;\n                        while ((parentNode = parentNode.parent)) {\n                            if (parentNode.contentType === 'multipart/related') {\n                                attachment.related = true;\n                            }\n                        }\n                    }\n\n                    attachment.headers = node.headers;\n                    this.push(attachment);\n                    this.attachmentList.push(attachment);\n                } else if (node.disposition === 'inline') {\n                    let chunks = [];\n                    let chunklen = 0;\n                    node.contentStream = node.decoder;\n\n                    if (node.contentType === 'text/plain') {\n                        this.hasText = true;\n                    } else if (node.contentType === 'text/html') {\n                        this.hasHtml = true;\n                    } else if (node.contentType === 'message/delivery-status' && !this.options.keepDeliveryStatus) {\n                        this.hasText = true;\n                    }\n\n                    if (node.node.flowed) {\n                        let contentStream = node.contentStream;\n                        let flowDecoder = new FlowedDecoder({\n                            delSp: node.node.delSp\n                        });\n                        contentStream.on('error', err => {\n                            flowDecoder.emit('error', err);\n                        });\n                        contentStream.pipe(flowDecoder);\n                        node.contentStream = flowDecoder;\n                    }\n\n                    let charset = node.charset || 'utf-8';\n                    //charset = charset || 'windows-1257';\n\n                    if (!['ascii', 'usascii', 'utf8'].includes(charset.toLowerCase().replace(/[^a-z0-9]+/g, ''))) {\n                        try {\n                            let contentStream = node.contentStream;\n                            let decodeStream = this.decoder.decodeStream(charset);\n                            contentStream.on('error', err => {\n                                decodeStream.emit('error', err);\n                            });\n                            contentStream.pipe(decodeStream);\n                            node.contentStream = decodeStream;\n                        } catch (E) {\n                            // do not decode charset\n                        }\n                    }\n\n                    node.contentStream.on('readable', () => {\n                        let chunk;\n                        while ((chunk = node.contentStream.read()) !== null) {\n                            if (typeof chunk === 'string') {\n                                chunk = Buffer.from(chunk);\n                            }\n                            chunks.push(chunk);\n                            chunklen += chunk.length;\n                        }\n                    });\n\n                    node.contentStream.once('end', () => {\n                        node.textContent = Buffer.concat(chunks, chunklen).toString().replace(/\\r?\\n/g, '\\n');\n                    });\n\n                    node.contentStream.once('error', err => {\n                        this.emit('error', err);\n                    });\n                }\n\n                break;\n            }\n\n            case 'data':\n                if (this.curnode && this.curnode.decoder) {\n                    this.curnode.decoder.end();\n                }\n\n                if (this.waitUntilAttachmentEnd) {\n                    this.attachmentCallback = done;\n                    return;\n                }\n\n                // multipart message structure\n                // this is not related to any specific 'node' block as it includes\n                // everything between the end of some node body and between the next header\n                //process.stdout.write(data.value);\n                break;\n\n            case 'body':\n                if (this.curnode && this.curnode.decoder && this.curnode.decoder.writable) {\n                    if (this.curnode.decoder.write(data.value) === false) {\n                        return this.curnode.decoder.once('drain', done);\n                    }\n                }\n\n                // Leaf element body. Includes the body for the last 'node' block. You might\n                // have several 'body' calls for a single 'node' block\n                //process.stdout.write(data.value);\n                break;\n        }\n\n        setImmediate(done);\n    }\n\n    _getPartId(parentBoundary) {\n        let boundaryIndex = this.boundaries.findIndex(item => item.name === parentBoundary);\n        if (boundaryIndex === -1) {\n            this.boundaries.push({ name: parentBoundary, count: 1 });\n            boundaryIndex = this.boundaries.length - 1;\n        } else {\n            this.boundaries[boundaryIndex].count++;\n        }\n        let partId = '1';\n        for (let i = 0; i <= boundaryIndex; i++) {\n            if (i === 0) partId = this.boundaries[i].count.toString();\n            else partId += '.' + this.boundaries[i].count.toString();\n        }\n        return partId;\n    }\n\n    getAddressesHTML(value) {\n        let formatSingleLevel = addresses =>\n            addresses\n                .map(address => {\n                    let str = '<span class=\"mp_address_group\">';\n                    if (address.name) {\n                        str += '<span class=\"mp_address_name\">' + he.encode(address.name) + (address.group ? ': ' : '') + '</span>';\n                    }\n                    if (address.address) {\n                        let link = '<a href=\"mailto:' + he.encode(address.address) + '\" class=\"mp_address_email\">' + he.encode(address.address) + '</a>';\n                        if (address.name) {\n                            str += ' &lt;' + link + '&gt;';\n                        } else {\n                            str += link;\n                        }\n                    }\n                    if (address.group) {\n                        str += formatSingleLevel(address.group) + ';';\n                    }\n                    return str + '</span>';\n                })\n                .join(', ');\n        return formatSingleLevel([].concat(value || []));\n    }\n\n    getAddressesText(value) {\n        let formatSingleLevel = addresses =>\n            addresses\n                .map(address => {\n                    let str = '';\n                    if (address.name) {\n                        str += `\"${address.name}\"` + (address.group ? ': ' : '');\n                    }\n                    if (address.address) {\n                        let link = address.address;\n                        if (address.name) {\n                            str += ' <' + link + '>';\n                        } else {\n                            str += link;\n                        }\n                    }\n                    if (address.group) {\n                        str += formatSingleLevel(address.group) + ';';\n                    }\n                    return str;\n                })\n                .join(', ');\n        return formatSingleLevel([].concat(value || []));\n    }\n\n    updateImageLinks(replaceCallback, done) {\n        if (!this.html) {\n            return setImmediate(() => done(null, false));\n        }\n\n        let cids = new Map();\n        let html = (this.html || '').toString();\n\n        if (this.options.skipImageLinks) {\n            return done(null, html);\n        }\n\n        html.replace(/\\bcid:([^'\"\\s]{1,256})/g, (match, cid) => {\n            for (let i = 0, len = this.attachmentList.length; i < len; i++) {\n                if (this.attachmentList[i].cid === cid && /^image\\/[\\w]+$/i.test(this.attachmentList[i].contentType)) {\n                    cids.set(cid, {\n                        attachment: this.attachmentList[i]\n                    });\n                    break;\n                }\n            }\n            return match;\n        });\n\n        let cidList = [];\n        cids.forEach(entry => {\n            cidList.push(entry);\n        });\n\n        let pos = 0;\n        let processNext = () => {\n            if (pos >= cidList.length) {\n                html = html.replace(/\\bcid:([^'\"\\s]{1,256})/g, (match, cid) => {\n                    if (cids.has(cid) && cids.get(cid).url) {\n                        return cids.get(cid).url;\n                    }\n                    return match;\n                });\n\n                return done(null, html);\n            }\n            let entry = cidList[pos++];\n            replaceCallback(entry.attachment, (err, url) => {\n                if (err) {\n                    return setImmediate(() => done(err));\n                }\n                entry.url = url;\n                setImmediate(processNext);\n            });\n        };\n\n        setImmediate(processNext);\n    }\n\n    textToHtml(str) {\n        if (this.options.skipTextToHtml) {\n            return '';\n        }\n        str = (str || '').toString();\n        let encoded;\n\n        let linkified = false;\n        if (!this.options.skipTextLinks) {\n            try {\n                if (linkify.pretest(str)) {\n                    linkified = true;\n                    let links = linkify.match(str) || [];\n                    let result = [];\n                    let last = 0;\n\n                    links.forEach(link => {\n                        if (last < link.index) {\n                            let textPart = he\n                                // encode special chars\n                                .encode(str.slice(last, link.index), {\n                                    useNamedReferences: true\n                                });\n                            result.push(textPart);\n                        }\n\n                        result.push(`<a href=\"${link.url}\">${link.text}</a>`);\n\n                        last = link.lastIndex;\n                    });\n\n                    let textPart = he\n                        // encode special chars\n                        .encode(str.slice(last), {\n                            useNamedReferences: true\n                        });\n                    result.push(textPart);\n\n                    encoded = result.join('');\n                }\n            } catch (E) {\n                // failed, don't linkify\n            }\n        }\n\n        if (!linkified) {\n            encoded = he\n                // encode special chars\n                .encode(str, {\n                    useNamedReferences: true\n                });\n        }\n\n        let text =\n            '<p>' +\n            encoded\n                .replace(/\\r?\\n/g, '\\n')\n                .trim() // normalize line endings\n                .replace(/[ \\t]+$/gm, '')\n                .trim() // trim empty line endings\n                .replace(/\\n\\n+/g, '</p><p>')\n                .trim() // insert <p> to multiple linebreaks\n                .replace(/\\n/g, '<br/>') + // insert <br> to single linebreaks\n            '</p>';\n\n        return text;\n    }\n}\n\nmodule.exports = MailParser;\n"],"names":[],"ignoreList":[0],"sourceRoot":""}\n//# sourceURL=webpack-internal:///(rsc)/./node_modules/mailparser/lib/mail-parser.js\n");
|
|
|
|
/***/ }),
|
|
|
|
/***/ "(rsc)/./node_modules/mailparser/lib/simple-parser.js":
|
|
/*!******************************************************!*\
|
|
!*** ./node_modules/mailparser/lib/simple-parser.js ***!
|
|
\******************************************************/
|
|
/***/ ((module, __unused_webpack_exports, __webpack_require__) => {
|
|
|
|
eval("\n\nconst MailParser = __webpack_require__(/*! ./mail-parser.js */ \"(rsc)/./node_modules/mailparser/lib/mail-parser.js\");\n\nmodule.exports = (input, options, callback) => {\n if (input === null || input === undefined) {\n throw new TypeError('Input cannot be null or undefined.');\n }\n\n if (!callback && typeof options === 'function') {\n callback = options;\n options = false;\n }\n\n let promise;\n if (!callback) {\n promise = new Promise((resolve, reject) => {\n callback = callbackPromise(resolve, reject);\n });\n }\n\n options = options || {};\n let keepCidLinks = !!options.keepCidLinks;\n\n let mail = {\n attachments: []\n };\n\n let parser = new MailParser(options);\n\n parser.on('error', err => {\n callback(err);\n });\n\n parser.on('headers', headers => {\n mail.headers = headers;\n mail.headerLines = parser.headerLines;\n });\n\n let reading = false;\n let reader = () => {\n reading = true;\n\n let data = parser.read();\n\n if (data === null) {\n reading = false;\n return;\n }\n\n if (data.type === 'text') {\n Object.keys(data).forEach(key => {\n if (['text', 'html', 'textAsHtml'].includes(key)) {\n mail[key] = data[key];\n }\n });\n }\n\n if (data.type === 'attachment') {\n mail.attachments.push(data);\n\n let chunks = [];\n let chunklen = 0;\n data.content.on('readable', () => {\n let chunk;\n while ((chunk = data.content.read()) !== null) {\n chunks.push(chunk);\n chunklen += chunk.length;\n }\n });\n\n data.content.on('end', () => {\n data.content = Buffer.concat(chunks, chunklen);\n data.release();\n reader();\n });\n } else {\n reader();\n }\n };\n\n parser.on('readable', () => {\n if (!reading) {\n reader();\n }\n });\n\n parser.on('end', () => {\n ['subject', 'references', 'date', 'to', 'from', 'to', 'cc', 'bcc', 'message-id', 'in-reply-to', 'reply-to'].forEach(key => {\n if (mail.headers && mail.headers.has(key)) {\n mail[key.replace(/-([a-z])/g, (m, c) => c.toUpperCase())] = mail.headers.get(key);\n }\n });\n\n if (keepCidLinks) {\n return callback(null, mail);\n }\n parser.updateImageLinks(\n (attachment, done) => done(false, 'data:' + attachment.contentType + ';base64,' + attachment.content.toString('base64')),\n (err, html) => {\n if (err) {\n return callback(err);\n }\n mail.html = html;\n\n callback(null, mail);\n }\n );\n });\n\n if (typeof input === 'string') {\n parser.end(Buffer.from(input));\n } else if (Buffer.isBuffer(input)) {\n parser.end(input);\n } else {\n input\n .once('error', err => {\n input.destroy();\n parser.destroy();\n callback(err);\n })\n .pipe(parser);\n }\n\n return promise;\n};\n\nfunction callbackPromise(resolve, reject) {\n return function (...args) {\n let err = args.shift();\n if (err) {\n reject(err);\n } else {\n resolve(...args);\n }\n };\n}\n//# sourceURL=[module]\n//# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiKHJzYykvLi9ub2RlX21vZHVsZXMvbWFpbHBhcnNlci9saWIvc2ltcGxlLXBhcnNlci5qcyIsIm1hcHBpbmdzIjoiQUFBYTs7QUFFYixtQkFBbUIsbUJBQU8sQ0FBQyw0RUFBa0I7O0FBRTdDO0FBQ0E7QUFDQTtBQUNBOztBQUVBO0FBQ0E7QUFDQTtBQUNBOztBQUVBO0FBQ0E7QUFDQTtBQUNBO0FBQ0EsU0FBUztBQUNUOztBQUVBO0FBQ0E7O0FBRUE7QUFDQTtBQUNBOztBQUVBOztBQUVBO0FBQ0E7QUFDQSxLQUFLOztBQUVMO0FBQ0E7QUFDQTtBQUNBLEtBQUs7O0FBRUw7QUFDQTtBQUNBOztBQUVBOztBQUVBO0FBQ0E7QUFDQTtBQUNBOztBQUVBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQSxhQUFhO0FBQ2I7O0FBRUE7QUFDQTs7QUFFQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0EsYUFBYTs7QUFFYjtBQUNBO0FBQ0E7QUFDQTtBQUNBLGFBQWE7QUFDYixVQUFVO0FBQ1Y7QUFDQTtBQUNBOztBQUVBO0FBQ0E7QUFDQTtBQUNBO0FBQ0EsS0FBSzs7QUFFTDtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0EsU0FBUzs7QUFFVDtBQUNBO0FBQ0E7QUFDQTtBQUNBLG1GQUFtRjtBQUNuRjtBQUNBO0FBQ0E7QUFDQTtBQUNBOztBQUVBO0FBQ0E7QUFDQTtBQUNBLEtBQUs7O0FBRUw7QUFDQTtBQUNBLE1BQU07QUFDTjtBQUNBLE1BQU07QUFDTjtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0EsYUFBYTtBQUNiO0FBQ0E7O0FBRUE7QUFDQTs7QUFFQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0EsVUFBVTtBQUNWO0FBQ0E7QUFDQTtBQUNBIiwic291cmNlcyI6WyIvaG9tZS9hbG1hL25leHRnZW4vTmVhaC1tYWlsL25vZGVfbW9kdWxlcy9tYWlscGFyc2VyL2xpYi9zaW1wbGUtcGFyc2VyLmpzIl0sInNvdXJjZXNDb250ZW50IjpbIid1c2Ugc3RyaWN0JztcblxuY29uc3QgTWFpbFBhcnNlciA9IHJlcXVpcmUoJy4vbWFpbC1wYXJzZXIuanMnKTtcblxubW9kdWxlLmV4cG9ydHMgPSAoaW5wdXQsIG9wdGlvbnMsIGNhbGxiYWNrKSA9PiB7XG4gICAgaWYgKGlucHV0ID09PSBudWxsIHx8IGlucHV0ID09PSB1bmRlZmluZWQpIHtcbiAgICAgICAgdGhyb3cgbmV3IFR5cGVFcnJvcignSW5wdXQgY2Fubm90IGJlIG51bGwgb3IgdW5kZWZpbmVkLicpO1xuICAgIH1cblxuICAgIGlmICghY2FsbGJhY2sgJiYgdHlwZW9mIG9wdGlvbnMgPT09ICdmdW5jdGlvbicpIHtcbiAgICAgICAgY2FsbGJhY2sgPSBvcHRpb25zO1xuICAgICAgICBvcHRpb25zID0gZmFsc2U7XG4gICAgfVxuXG4gICAgbGV0IHByb21pc2U7XG4gICAgaWYgKCFjYWxsYmFjaykge1xuICAgICAgICBwcm9taXNlID0gbmV3IFByb21pc2UoKHJlc29sdmUsIHJlamVjdCkgPT4ge1xuICAgICAgICAgICAgY2FsbGJhY2sgPSBjYWxsYmFja1Byb21pc2UocmVzb2x2ZSwgcmVqZWN0KTtcbiAgICAgICAgfSk7XG4gICAgfVxuXG4gICAgb3B0aW9ucyA9IG9wdGlvbnMgfHwge307XG4gICAgbGV0IGtlZXBDaWRMaW5rcyA9ICEhb3B0aW9ucy5rZWVwQ2lkTGlua3M7XG5cbiAgICBsZXQgbWFpbCA9IHtcbiAgICAgICAgYXR0YWNobWVudHM6IFtdXG4gICAgfTtcblxuICAgIGxldCBwYXJzZXIgPSBuZXcgTWFpbFBhcnNlcihvcHRpb25zKTtcblxuICAgIHBhcnNlci5vbignZXJyb3InLCBlcnIgPT4ge1xuICAgICAgICBjYWxsYmFjayhlcnIpO1xuICAgIH0pO1xuXG4gICAgcGFyc2VyLm9uKCdoZWFkZXJzJywgaGVhZGVycyA9PiB7XG4gICAgICAgIG1haWwuaGVhZGVycyA9IGhlYWRlcnM7XG4gICAgICAgIG1haWwuaGVhZGVyTGluZXMgPSBwYXJzZXIuaGVhZGVyTGluZXM7XG4gICAgfSk7XG5cbiAgICBsZXQgcmVhZGluZyA9IGZhbHNlO1xuICAgIGxldCByZWFkZXIgPSAoKSA9PiB7XG4gICAgICAgIHJlYWRpbmcgPSB0cnVlO1xuXG4gICAgICAgIGxldCBkYXRhID0gcGFyc2VyLnJlYWQoKTtcblxuICAgICAgICBpZiAoZGF0YSA9PT0gbnVsbCkge1xuICAgICAgICAgICAgcmVhZGluZyA9IGZhbHNlO1xuICAgICAgICAgICAgcmV0dXJuO1xuICAgICAgICB9XG5cbiAgICAgICAgaWYgKGRhdGEudHlwZSA9PT0gJ3RleHQnKSB7XG4gICAgICAgICAgICBPYmplY3Qua2V5cyhkYXRhKS5mb3JFYWNoKGtleSA9PiB7XG4gICAgICAgICAgICAgICAgaWYgKFsndGV4dCcsICdodG1sJywgJ3RleHRBc0h0bWwnXS5pbmNsdWRlcyhrZXkpKSB7XG4gICAgICAgICAgICAgICAgICAgIG1haWxba2V5XSA9IGRhdGFba2V5XTtcbiAgICAgICAgICAgICAgICB9XG4gICAgICAgICAgICB9KTtcbiAgICAgICAgfVxuXG4gICAgICAgIGlmIChkYXRhLnR5cGUgPT09ICdhdHRhY2htZW50Jykge1xuICAgICAgICAgICAgbWFpbC5hdHRhY2htZW50cy5wdXNoKGRhdGEpO1xuXG4gICAgICAgICAgICBsZXQgY2h1bmtzID0gW107XG4gICAgICAgICAgICBsZXQgY2h1bmtsZW4gPSAwO1xuICAgICAgICAgICAgZGF0YS5jb250ZW50Lm9uKCdyZWFkYWJsZScsICgpID0+IHtcbiAgICAgICAgICAgICAgICBsZXQgY2h1bms7XG4gICAgICAgICAgICAgICAgd2hpbGUgKChjaHVuayA9IGRhdGEuY29udGVudC5yZWFkKCkpICE9PSBudWxsKSB7XG4gICAgICAgICAgICAgICAgICAgIGNodW5rcy5wdXNoKGNodW5rKTtcbiAgICAgICAgICAgICAgICAgICAgY2h1bmtsZW4gKz0gY2h1bmsubGVuZ3RoO1xuICAgICAgICAgICAgICAgIH1cbiAgICAgICAgICAgIH0pO1xuXG4gICAgICAgICAgICBkYXRhLmNvbnRlbnQub24oJ2VuZCcsICgpID0+IHtcbiAgICAgICAgICAgICAgICBkYXRhLmNvbnRlbnQgPSBCdWZmZXIuY29uY2F0KGNodW5rcywgY2h1bmtsZW4pO1xuICAgICAgICAgICAgICAgIGRhdGEucmVsZWFzZSgpO1xuICAgICAgICAgICAgICAgIHJlYWRlcigpO1xuICAgICAgICAgICAgfSk7XG4gICAgICAgIH0gZWxzZSB7XG4gICAgICAgICAgICByZWFkZXIoKTtcbiAgICAgICAgfVxuICAgIH07XG5cbiAgICBwYXJzZXIub24oJ3JlYWRhYmxlJywgKCkgPT4ge1xuICAgICAgICBpZiAoIXJlYWRpbmcpIHtcbiAgICAgICAgICAgIHJlYWRlcigpO1xuICAgICAgICB9XG4gICAgfSk7XG5cbiAgICBwYXJzZXIub24oJ2VuZCcsICgpID0+IHtcbiAgICAgICAgWydzdWJqZWN0JywgJ3JlZmVyZW5jZXMnLCAnZGF0ZScsICd0bycsICdmcm9tJywgJ3RvJywgJ2NjJywgJ2JjYycsICdtZXNzYWdlLWlkJywgJ2luLXJlcGx5LXRvJywgJ3JlcGx5LXRvJ10uZm9yRWFjaChrZXkgPT4ge1xuICAgICAgICAgICAgaWYgKG1haWwuaGVhZGVycyAmJiBtYWlsLmhlYWRlcnMuaGFzKGtleSkpIHtcbiAgICAgICAgICAgICAgICBtYWlsW2tleS5yZXBsYWNlKC8tKFthLXpdKS9nLCAobSwgYykgPT4gYy50b1VwcGVyQ2FzZSgpKV0gPSBtYWlsLmhlYWRlcnMuZ2V0KGtleSk7XG4gICAgICAgICAgICB9XG4gICAgICAgIH0pO1xuXG4gICAgICAgIGlmIChrZWVwQ2lkTGlua3MpIHtcbiAgICAgICAgICAgIHJldHVybiBjYWxsYmFjayhudWxsLCBtYWlsKTtcbiAgICAgICAgfVxuICAgICAgICBwYXJzZXIudXBkYXRlSW1hZ2VMaW5rcyhcbiAgICAgICAgICAgIChhdHRhY2htZW50LCBkb25lKSA9PiBkb25lKGZhbHNlLCAnZGF0YTonICsgYXR0YWNobWVudC5jb250ZW50VHlwZSArICc7YmFzZTY0LCcgKyBhdHRhY2htZW50LmNvbnRlbnQudG9TdHJpbmcoJ2Jhc2U2NCcpKSxcbiAgICAgICAgICAgIChlcnIsIGh0bWwpID0+IHtcbiAgICAgICAgICAgICAgICBpZiAoZXJyKSB7XG4gICAgICAgICAgICAgICAgICAgIHJldHVybiBjYWxsYmFjayhlcnIpO1xuICAgICAgICAgICAgICAgIH1cbiAgICAgICAgICAgICAgICBtYWlsLmh0bWwgPSBodG1sO1xuXG4gICAgICAgICAgICAgICAgY2FsbGJhY2sobnVsbCwgbWFpbCk7XG4gICAgICAgICAgICB9XG4gICAgICAgICk7XG4gICAgfSk7XG5cbiAgICBpZiAodHlwZW9mIGlucHV0ID09PSAnc3RyaW5nJykge1xuICAgICAgICBwYXJzZXIuZW5kKEJ1ZmZlci5mcm9tKGlucHV0KSk7XG4gICAgfSBlbHNlIGlmIChCdWZmZXIuaXNCdWZmZXIoaW5wdXQpKSB7XG4gICAgICAgIHBhcnNlci5lbmQoaW5wdXQpO1xuICAgIH0gZWxzZSB7XG4gICAgICAgIGlucHV0XG4gICAgICAgICAgICAub25jZSgnZXJyb3InLCBlcnIgPT4ge1xuICAgICAgICAgICAgICAgIGlucHV0LmRlc3Ryb3koKTtcbiAgICAgICAgICAgICAgICBwYXJzZXIuZGVzdHJveSgpO1xuICAgICAgICAgICAgICAgIGNhbGxiYWNrKGVycik7XG4gICAgICAgICAgICB9KVxuICAgICAgICAgICAgLnBpcGUocGFyc2VyKTtcbiAgICB9XG5cbiAgICByZXR1cm4gcHJvbWlzZTtcbn07XG5cbmZ1bmN0aW9uIGNhbGxiYWNrUHJvbWlzZShyZXNvbHZlLCByZWplY3QpIHtcbiAgICByZXR1cm4gZnVuY3Rpb24gKC4uLmFyZ3MpIHtcbiAgICAgICAgbGV0IGVyciA9IGFyZ3Muc2hpZnQoKTtcbiAgICAgICAgaWYgKGVycikge1xuICAgICAgICAgICAgcmVqZWN0KGVycik7XG4gICAgICAgIH0gZWxzZSB7XG4gICAgICAgICAgICByZXNvbHZlKC4uLmFyZ3MpO1xuICAgICAgICB9XG4gICAgfTtcbn1cbiJdLCJuYW1lcyI6W10sImlnbm9yZUxpc3QiOlswXSwic291cmNlUm9vdCI6IiJ9\n//# sourceURL=webpack-internal:///(rsc)/./node_modules/mailparser/lib/simple-parser.js\n");
|
|
|
|
/***/ }),
|
|
|
|
/***/ "(rsc)/./node_modules/mailparser/lib/stream-hash.js":
|
|
/*!****************************************************!*\
|
|
!*** ./node_modules/mailparser/lib/stream-hash.js ***!
|
|
\****************************************************/
|
|
/***/ ((module, __unused_webpack_exports, __webpack_require__) => {
|
|
|
|
eval("\n\nconst crypto = __webpack_require__(/*! crypto */ \"crypto\");\nconst Transform = (__webpack_require__(/*! stream */ \"stream\").Transform);\n\nclass StreamHash extends Transform {\n constructor(attachment, algo) {\n super();\n this.attachment = attachment;\n this.algo = (algo || 'md5').toLowerCase();\n this.hash = crypto.createHash(algo);\n this.byteCount = 0;\n }\n\n _transform(chunk, encoding, done) {\n this.hash.update(chunk);\n this.byteCount += chunk.length;\n done(null, chunk);\n }\n\n _flush(done) {\n this.attachment.checksum = this.hash.digest('hex');\n this.attachment.size = this.byteCount;\n done();\n }\n}\n\nmodule.exports = StreamHash;\n//# sourceURL=[module]\n//# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiKHJzYykvLi9ub2RlX21vZHVsZXMvbWFpbHBhcnNlci9saWIvc3RyZWFtLWhhc2guanMiLCJtYXBwaW5ncyI6IkFBQWE7O0FBRWIsZUFBZSxtQkFBTyxDQUFDLHNCQUFRO0FBQy9CLGtCQUFrQix1REFBMkI7O0FBRTdDO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7O0FBRUE7QUFDQTtBQUNBO0FBQ0E7QUFDQTs7QUFFQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7O0FBRUEiLCJzb3VyY2VzIjpbIi9ob21lL2FsbWEvbmV4dGdlbi9OZWFoLW1haWwvbm9kZV9tb2R1bGVzL21haWxwYXJzZXIvbGliL3N0cmVhbS1oYXNoLmpzIl0sInNvdXJjZXNDb250ZW50IjpbIid1c2Ugc3RyaWN0JztcblxuY29uc3QgY3J5cHRvID0gcmVxdWlyZSgnY3J5cHRvJyk7XG5jb25zdCBUcmFuc2Zvcm0gPSByZXF1aXJlKCdzdHJlYW0nKS5UcmFuc2Zvcm07XG5cbmNsYXNzIFN0cmVhbUhhc2ggZXh0ZW5kcyBUcmFuc2Zvcm0ge1xuICAgIGNvbnN0cnVjdG9yKGF0dGFjaG1lbnQsIGFsZ28pIHtcbiAgICAgICAgc3VwZXIoKTtcbiAgICAgICAgdGhpcy5hdHRhY2htZW50ID0gYXR0YWNobWVudDtcbiAgICAgICAgdGhpcy5hbGdvID0gKGFsZ28gfHwgJ21kNScpLnRvTG93ZXJDYXNlKCk7XG4gICAgICAgIHRoaXMuaGFzaCA9IGNyeXB0by5jcmVhdGVIYXNoKGFsZ28pO1xuICAgICAgICB0aGlzLmJ5dGVDb3VudCA9IDA7XG4gICAgfVxuXG4gICAgX3RyYW5zZm9ybShjaHVuaywgZW5jb2RpbmcsIGRvbmUpIHtcbiAgICAgICAgdGhpcy5oYXNoLnVwZGF0ZShjaHVuayk7XG4gICAgICAgIHRoaXMuYnl0ZUNvdW50ICs9IGNodW5rLmxlbmd0aDtcbiAgICAgICAgZG9uZShudWxsLCBjaHVuayk7XG4gICAgfVxuXG4gICAgX2ZsdXNoKGRvbmUpIHtcbiAgICAgICAgdGhpcy5hdHRhY2htZW50LmNoZWNrc3VtID0gdGhpcy5oYXNoLmRpZ2VzdCgnaGV4Jyk7XG4gICAgICAgIHRoaXMuYXR0YWNobWVudC5zaXplID0gdGhpcy5ieXRlQ291bnQ7XG4gICAgICAgIGRvbmUoKTtcbiAgICB9XG59XG5cbm1vZHVsZS5leHBvcnRzID0gU3RyZWFtSGFzaDtcbiJdLCJuYW1lcyI6W10sImlnbm9yZUxpc3QiOlswXSwic291cmNlUm9vdCI6IiJ9\n//# sourceURL=webpack-internal:///(rsc)/./node_modules/mailparser/lib/stream-hash.js\n");
|
|
|
|
/***/ }),
|
|
|
|
/***/ "(rsc)/./node_modules/mailparser/node_modules/nodemailer/lib/addressparser/index.js":
|
|
/*!************************************************************************************!*\
|
|
!*** ./node_modules/mailparser/node_modules/nodemailer/lib/addressparser/index.js ***!
|
|
\************************************************************************************/
|
|
/***/ ((module) => {
|
|
|
|
eval("\n\n/**\n * Converts tokens for a single address into an address object\n *\n * @param {Array} tokens Tokens object\n * @return {Object} Address object\n */\nfunction _handleAddress(tokens) {\n let isGroup = false;\n let state = 'text';\n let address;\n let addresses = [];\n let data = {\n address: [],\n comment: [],\n group: [],\n text: []\n };\n let i;\n let len;\n\n // Filter out <addresses>, (comments) and regular text\n for (i = 0, len = tokens.length; i < len; i++) {\n let token = tokens[i];\n let prevToken = i ? tokens[i - 1] : null;\n if (token.type === 'operator') {\n switch (token.value) {\n case '<':\n state = 'address';\n break;\n case '(':\n state = 'comment';\n break;\n case ':':\n state = 'group';\n isGroup = true;\n break;\n default:\n state = 'text';\n break;\n }\n } else if (token.value) {\n if (state === 'address') {\n // handle use case where unquoted name includes a \"<\"\n // Apple Mail truncates everything between an unexpected < and an address\n // and so will we\n token.value = token.value.replace(/^[^<]*<\\s*/, '');\n }\n\n if (prevToken && prevToken.noBreak && data[state].length) {\n // join values\n data[state][data[state].length - 1] += token.value;\n } else {\n data[state].push(token.value);\n }\n }\n }\n\n // If there is no text but a comment, replace the two\n if (!data.text.length && data.comment.length) {\n data.text = data.comment;\n data.comment = [];\n }\n\n if (isGroup) {\n // http://tools.ietf.org/html/rfc2822#appendix-A.1.3\n data.text = data.text.join(' ');\n addresses.push({\n name: data.text || (address && address.name),\n group: data.group.length ? addressparser(data.group.join(',')) : []\n });\n } else {\n // If no address was found, try to detect one from regular text\n if (!data.address.length && data.text.length) {\n for (i = data.text.length - 1; i >= 0; i--) {\n if (data.text[i].match(/^[^@\\s]+@[^@\\s]+$/)) {\n data.address = data.text.splice(i, 1);\n break;\n }\n }\n\n let _regexHandler = function (address) {\n if (!data.address.length) {\n data.address = [address.trim()];\n return ' ';\n } else {\n return address;\n }\n };\n\n // still no address\n if (!data.address.length) {\n for (i = data.text.length - 1; i >= 0; i--) {\n // fixed the regex to parse email address correctly when email address has more than one @\n data.text[i] = data.text[i].replace(/\\s*\\b[^@\\s]+@[^\\s]+\\b\\s*/, _regexHandler).trim();\n if (data.address.length) {\n break;\n }\n }\n }\n }\n\n // If there's still is no text but a comment exixts, replace the two\n if (!data.text.length && data.comment.length) {\n data.text = data.comment;\n data.comment = [];\n }\n\n // Keep only the first address occurence, push others to regular text\n if (data.address.length > 1) {\n data.text = data.text.concat(data.address.splice(1));\n }\n\n // Join values with spaces\n data.text = data.text.join(' ');\n data.address = data.address.join(' ');\n\n if (!data.address && isGroup) {\n return [];\n } else {\n address = {\n address: data.address || data.text || '',\n name: data.text || data.address || ''\n };\n\n if (address.address === address.name) {\n if ((address.address || '').match(/@/)) {\n address.name = '';\n } else {\n address.address = '';\n }\n }\n\n addresses.push(address);\n }\n }\n\n return addresses;\n}\n\n/**\n * Creates a Tokenizer object for tokenizing address field strings\n *\n * @constructor\n * @param {String} str Address field string\n */\nclass Tokenizer {\n constructor(str) {\n this.str = (str || '').toString();\n this.operatorCurrent = '';\n this.operatorExpecting = '';\n this.node = null;\n this.escaped = false;\n\n this.list = [];\n /**\n * Operator tokens and which tokens are expected to end the sequence\n */\n this.operators = {\n '\"': '\"',\n '(': ')',\n '<': '>',\n ',': '',\n ':': ';',\n // Semicolons are not a legal delimiter per the RFC2822 grammar other\n // than for terminating a group, but they are also not valid for any\n // other use in this context. Given that some mail clients have\n // historically allowed the semicolon as a delimiter equivalent to the\n // comma in their UI, it makes sense to treat them the same as a comma\n // when used outside of a group.\n ';': ''\n };\n }\n\n /**\n * Tokenizes the original input string\n *\n * @return {Array} An array of operator|text tokens\n */\n tokenize() {\n let list = [];\n\n for (let i = 0, len = this.str.length; i < len; i++) {\n let chr = this.str.charAt(i);\n let nextChr = i < len - 1 ? this.str.charAt(i + 1) : null;\n this.checkChar(chr, nextChr);\n }\n\n this.list.forEach(node => {\n node.value = (node.value || '').toString().trim();\n if (node.value) {\n list.push(node);\n }\n });\n\n return list;\n }\n\n /**\n * Checks if a character is an operator or text and acts accordingly\n *\n * @param {String} chr Character from the address field\n */\n checkChar(chr, nextChr) {\n if (this.escaped) {\n // ignore next condition blocks\n } else if (chr === this.operatorExpecting) {\n this.node = {\n type: 'operator',\n value: chr\n };\n\n if (nextChr && ![' ', '\\t', '\\r', '\\n', ',', ';'].includes(nextChr)) {\n this.node.noBreak = true;\n }\n\n this.list.push(this.node);\n this.node = null;\n this.operatorExpecting = '';\n this.escaped = false;\n\n return;\n } else if (!this.operatorExpecting && chr in this.operators) {\n this.node = {\n type: 'operator',\n value: chr\n };\n this.list.push(this.node);\n this.node = null;\n this.operatorExpecting = this.operators[chr];\n this.escaped = false;\n return;\n } else if (['\"', \"'\"].includes(this.operatorExpecting) && chr === '\\\\') {\n this.escaped = true;\n return;\n }\n\n if (!this.node) {\n this.node = {\n type: 'text',\n value: ''\n };\n this.list.push(this.node);\n }\n\n if (chr === '\\n') {\n // Convert newlines to spaces. Carriage return is ignored as \\r and \\n usually\n // go together anyway and there already is a WS for \\n. Lone \\r means something is fishy.\n chr = ' ';\n }\n\n if (chr.charCodeAt(0) >= 0x21 || [' ', '\\t'].includes(chr)) {\n // skip command bytes\n this.node.value += chr;\n }\n\n this.escaped = false;\n }\n}\n\n/**\n * Parses structured e-mail addresses from an address field\n *\n * Example:\n *\n * 'Name <address@domain>'\n *\n * will be converted to\n *\n * [{name: 'Name', address: 'address@domain'}]\n *\n * @param {String} str Address field\n * @return {Array} An array of address objects\n */\nfunction addressparser(str, options) {\n options = options || {};\n\n let tokenizer = new Tokenizer(str);\n let tokens = tokenizer.tokenize();\n\n let addresses = [];\n let address = [];\n let parsedAddresses = [];\n\n tokens.forEach(token => {\n if (token.type === 'operator' && (token.value === ',' || token.value === ';')) {\n if (address.length) {\n addresses.push(address);\n }\n address = [];\n } else {\n address.push(token);\n }\n });\n\n if (address.length) {\n addresses.push(address);\n }\n\n addresses.forEach(address => {\n address = _handleAddress(address);\n if (address.length) {\n parsedAddresses = parsedAddresses.concat(address);\n }\n });\n\n if (options.flatten) {\n let addresses = [];\n let walkAddressList = list => {\n list.forEach(address => {\n if (address.group) {\n return walkAddressList(address.group);\n } else {\n addresses.push(address);\n }\n });\n };\n walkAddressList(parsedAddresses);\n return addresses;\n }\n\n return parsedAddresses;\n}\n\n// expose to the world\nmodule.exports = addressparser;\n//# sourceURL=[module]\n//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"file":"(rsc)/./node_modules/mailparser/node_modules/nodemailer/lib/addressparser/index.js","mappings":"AAAa;;AAEb;AACA;AACA;AACA,WAAW,OAAO;AAClB,YAAY,QAAQ;AACpB;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA,qCAAqC,SAAS;AAC9C;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,UAAU;AACV;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA,cAAc;AACd;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA,SAAS;AACT,MAAM;AACN;AACA;AACA,2CAA2C,QAAQ;AACnD;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA,kBAAkB;AAClB;AACA;AACA;;AAEA;AACA;AACA,+CAA+C,QAAQ;AACvD;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;;AAEA;AACA;AACA;;AAEA;AACA;AACA,UAAU;AACV;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA,kBAAkB;AAClB;AACA;AACA;;AAEA;AACA;AACA;;AAEA;AACA;;AAEA;AACA;AACA;AACA;AACA,WAAW,QAAQ;AACnB;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,mBAAmB;AACnB;AACA;AACA;AACA;AACA;AACA;AACA,cAAc;AACd;AACA;;AAEA;AACA;AACA;AACA,gBAAgB,OAAO;AACvB;AACA;AACA;;AAEA,+CAA+C,SAAS;AACxD;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA,SAAS;;AAET;AACA;;AAEA;AACA;AACA;AACA,eAAe,QAAQ;AACvB;AACA;AACA;AACA;AACA,UAAU;AACV;AACA;AACA;AACA;;AAEA,2DAA2D;AAC3D;AACA;;AAEA;AACA;AACA;AACA;;AAEA;AACA,UAAU;AACV;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,UAAU;AACV;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;;AAEA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,SAAS,wCAAwC;AACjD;AACA,WAAW,QAAQ;AACnB,YAAY,OAAO;AACnB;AACA;AACA;;AAEA;AACA;;AAEA;AACA;AACA;;AAEA;AACA,mFAAmF;AACnF;AACA;AACA;AACA;AACA,UAAU;AACV;AACA;AACA,KAAK;;AAEL;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA,KAAK;;AAEL;AACA;AACA;AACA;AACA;AACA;AACA,kBAAkB;AAClB;AACA;AACA,aAAa;AACb;AACA;AACA;AACA;;AAEA;AACA;;AAEA;AACA","sources":["/home/alma/nextgen/Neah-mail/node_modules/mailparser/node_modules/nodemailer/lib/addressparser/index.js"],"sourcesContent":["'use strict';\n\n/**\n * Converts tokens for a single address into an address object\n *\n * @param {Array} tokens Tokens object\n * @return {Object} Address object\n */\nfunction _handleAddress(tokens) {\n    let isGroup = false;\n    let state = 'text';\n    let address;\n    let addresses = [];\n    let data = {\n        address: [],\n        comment: [],\n        group: [],\n        text: []\n    };\n    let i;\n    let len;\n\n    // Filter out <addresses>, (comments) and regular text\n    for (i = 0, len = tokens.length; i < len; i++) {\n        let token = tokens[i];\n        let prevToken = i ? tokens[i - 1] : null;\n        if (token.type === 'operator') {\n            switch (token.value) {\n                case '<':\n                    state = 'address';\n                    break;\n                case '(':\n                    state = 'comment';\n                    break;\n                case ':':\n                    state = 'group';\n                    isGroup = true;\n                    break;\n                default:\n                    state = 'text';\n                    break;\n            }\n        } else if (token.value) {\n            if (state === 'address') {\n                // handle use case where unquoted name includes a \"<\"\n                // Apple Mail truncates everything between an unexpected < and an address\n                // and so will we\n                token.value = token.value.replace(/^[^<]*<\\s*/, '');\n            }\n\n            if (prevToken && prevToken.noBreak && data[state].length) {\n                // join values\n                data[state][data[state].length - 1] += token.value;\n            } else {\n                data[state].push(token.value);\n            }\n        }\n    }\n\n    // If there is no text but a comment, replace the two\n    if (!data.text.length && data.comment.length) {\n        data.text = data.comment;\n        data.comment = [];\n    }\n\n    if (isGroup) {\n        // http://tools.ietf.org/html/rfc2822#appendix-A.1.3\n        data.text = data.text.join(' ');\n        addresses.push({\n            name: data.text || (address && address.name),\n            group: data.group.length ? addressparser(data.group.join(',')) : []\n        });\n    } else {\n        // If no address was found, try to detect one from regular text\n        if (!data.address.length && data.text.length) {\n            for (i = data.text.length - 1; i >= 0; i--) {\n                if (data.text[i].match(/^[^@\\s]+@[^@\\s]+$/)) {\n                    data.address = data.text.splice(i, 1);\n                    break;\n                }\n            }\n\n            let _regexHandler = function (address) {\n                if (!data.address.length) {\n                    data.address = [address.trim()];\n                    return ' ';\n                } else {\n                    return address;\n                }\n            };\n\n            // still no address\n            if (!data.address.length) {\n                for (i = data.text.length - 1; i >= 0; i--) {\n                    // fixed the regex to parse email address correctly when email address has more than one @\n                    data.text[i] = data.text[i].replace(/\\s*\\b[^@\\s]+@[^\\s]+\\b\\s*/, _regexHandler).trim();\n                    if (data.address.length) {\n                        break;\n                    }\n                }\n            }\n        }\n\n        // If there's still is no text but a comment exixts, replace the two\n        if (!data.text.length && data.comment.length) {\n            data.text = data.comment;\n            data.comment = [];\n        }\n\n        // Keep only the first address occurence, push others to regular text\n        if (data.address.length > 1) {\n            data.text = data.text.concat(data.address.splice(1));\n        }\n\n        // Join values with spaces\n        data.text = data.text.join(' ');\n        data.address = data.address.join(' ');\n\n        if (!data.address && isGroup) {\n            return [];\n        } else {\n            address = {\n                address: data.address || data.text || '',\n                name: data.text || data.address || ''\n            };\n\n            if (address.address === address.name) {\n                if ((address.address || '').match(/@/)) {\n                    address.name = '';\n                } else {\n                    address.address = '';\n                }\n            }\n\n            addresses.push(address);\n        }\n    }\n\n    return addresses;\n}\n\n/**\n * Creates a Tokenizer object for tokenizing address field strings\n *\n * @constructor\n * @param {String} str Address field string\n */\nclass Tokenizer {\n    constructor(str) {\n        this.str = (str || '').toString();\n        this.operatorCurrent = '';\n        this.operatorExpecting = '';\n        this.node = null;\n        this.escaped = false;\n\n        this.list = [];\n        /**\n         * Operator tokens and which tokens are expected to end the sequence\n         */\n        this.operators = {\n            '\"': '\"',\n            '(': ')',\n            '<': '>',\n            ',': '',\n            ':': ';',\n            // Semicolons are not a legal delimiter per the RFC2822 grammar other\n            // than for terminating a group, but they are also not valid for any\n            // other use in this context.  Given that some mail clients have\n            // historically allowed the semicolon as a delimiter equivalent to the\n            // comma in their UI, it makes sense to treat them the same as a comma\n            // when used outside of a group.\n            ';': ''\n        };\n    }\n\n    /**\n     * Tokenizes the original input string\n     *\n     * @return {Array} An array of operator|text tokens\n     */\n    tokenize() {\n        let list = [];\n\n        for (let i = 0, len = this.str.length; i < len; i++) {\n            let chr = this.str.charAt(i);\n            let nextChr = i < len - 1 ? this.str.charAt(i + 1) : null;\n            this.checkChar(chr, nextChr);\n        }\n\n        this.list.forEach(node => {\n            node.value = (node.value || '').toString().trim();\n            if (node.value) {\n                list.push(node);\n            }\n        });\n\n        return list;\n    }\n\n    /**\n     * Checks if a character is an operator or text and acts accordingly\n     *\n     * @param {String} chr Character from the address field\n     */\n    checkChar(chr, nextChr) {\n        if (this.escaped) {\n            // ignore next condition blocks\n        } else if (chr === this.operatorExpecting) {\n            this.node = {\n                type: 'operator',\n                value: chr\n            };\n\n            if (nextChr && ![' ', '\\t', '\\r', '\\n', ',', ';'].includes(nextChr)) {\n                this.node.noBreak = true;\n            }\n\n            this.list.push(this.node);\n            this.node = null;\n            this.operatorExpecting = '';\n            this.escaped = false;\n\n            return;\n        } else if (!this.operatorExpecting && chr in this.operators) {\n            this.node = {\n                type: 'operator',\n                value: chr\n            };\n            this.list.push(this.node);\n            this.node = null;\n            this.operatorExpecting = this.operators[chr];\n            this.escaped = false;\n            return;\n        } else if (['\"', \"'\"].includes(this.operatorExpecting) && chr === '\\\\') {\n            this.escaped = true;\n            return;\n        }\n\n        if (!this.node) {\n            this.node = {\n                type: 'text',\n                value: ''\n            };\n            this.list.push(this.node);\n        }\n\n        if (chr === '\\n') {\n            // Convert newlines to spaces. Carriage return is ignored as \\r and \\n usually\n            // go together anyway and there already is a WS for \\n. Lone \\r means something is fishy.\n            chr = ' ';\n        }\n\n        if (chr.charCodeAt(0) >= 0x21 || [' ', '\\t'].includes(chr)) {\n            // skip command bytes\n            this.node.value += chr;\n        }\n\n        this.escaped = false;\n    }\n}\n\n/**\n * Parses structured e-mail addresses from an address field\n *\n * Example:\n *\n *    'Name <address@domain>'\n *\n * will be converted to\n *\n *     [{name: 'Name', address: 'address@domain'}]\n *\n * @param {String} str Address field\n * @return {Array} An array of address objects\n */\nfunction addressparser(str, options) {\n    options = options || {};\n\n    let tokenizer = new Tokenizer(str);\n    let tokens = tokenizer.tokenize();\n\n    let addresses = [];\n    let address = [];\n    let parsedAddresses = [];\n\n    tokens.forEach(token => {\n        if (token.type === 'operator' && (token.value === ',' || token.value === ';')) {\n            if (address.length) {\n                addresses.push(address);\n            }\n            address = [];\n        } else {\n            address.push(token);\n        }\n    });\n\n    if (address.length) {\n        addresses.push(address);\n    }\n\n    addresses.forEach(address => {\n        address = _handleAddress(address);\n        if (address.length) {\n            parsedAddresses = parsedAddresses.concat(address);\n        }\n    });\n\n    if (options.flatten) {\n        let addresses = [];\n        let walkAddressList = list => {\n            list.forEach(address => {\n                if (address.group) {\n                    return walkAddressList(address.group);\n                } else {\n                    addresses.push(address);\n                }\n            });\n        };\n        walkAddressList(parsedAddresses);\n        return addresses;\n    }\n\n    return parsedAddresses;\n}\n\n// expose to the world\nmodule.exports = addressparser;\n"],"names":[],"ignoreList":[0],"sourceRoot":""}\n//# sourceURL=webpack-internal:///(rsc)/./node_modules/mailparser/node_modules/nodemailer/lib/addressparser/index.js\n");
|
|
|
|
/***/ })
|
|
|
|
};
|
|
; |