UNPKG

emailjs-imap-client

Version:
233 lines (202 loc) 22.3 kB
"use strict"; Object.defineProperty(exports, "__esModule", { value: true }); exports.buildFETCHCommand = buildFETCHCommand; exports.buildXOAuth2Token = buildXOAuth2Token; exports.buildSEARCHCommand = buildSEARCHCommand; exports.buildSTORECommand = buildSTORECommand; var _emailjsImapHandler = require("emailjs-imap-handler"); var _emailjsMimeCodec = require("emailjs-mime-codec"); var _emailjsBase = require("emailjs-base64"); var _common = require("./common"); function buildFETCHCommand(sequence, items, options) { const command = { command: options.byUid ? 'UID FETCH' : 'FETCH', attributes: [{ type: 'SEQUENCE', value: sequence }] }; if (options.valueAsString !== undefined) { command.valueAsString = options.valueAsString; } let query = []; items.forEach(item => { item = item.toUpperCase().trim(); if (/^\w+$/.test(item)) { // alphanum strings can be used directly query.push({ type: 'ATOM', value: item }); } else if (item) { try { // parse the value as a fake command, use only the attributes block const cmd = (0, _emailjsImapHandler.parser)((0, _common.toTypedArray)('* Z ' + item)); query = query.concat(cmd.attributes || []); } catch (e) { // if parse failed, use the original string as one entity query.push({ type: 'ATOM', value: item }); } } }); if (query.length === 1) { query = query.pop(); } command.attributes.push(query); if (options.changedSince) { command.attributes.push([{ type: 'ATOM', value: 'CHANGEDSINCE' }, { type: 'ATOM', value: options.changedSince }]); } return command; } /** * Builds a login token for XOAUTH2 authentication command * * @param {String} user E-mail address of the user * @param {String} token Valid access token for the user * @return {String} Base64 formatted login token */ function buildXOAuth2Token(user = '', token) { const authData = [`user=${user}`, `auth=Bearer ${token}`, '', '']; return (0, _emailjsBase.encode)(authData.join('\x01')); } /** * Compiles a search query into an IMAP command. Queries are composed as objects * where keys are search terms and values are term arguments. Only strings, * numbers and Dates are used. If the value is an array, the members of it * are processed separately (use this for terms that require multiple params). * If the value is a Date, it is converted to the form of "01-Jan-1970". * Subqueries (OR, NOT) are made up of objects * * {unseen: true, header: ["subject", "hello world"]}; * SEARCH UNSEEN HEADER "subject" "hello world" * * @param {Object} query Search query * @param {Object} [options] Option object * @param {Boolean} [options.byUid] If ture, use UID SEARCH instead of SEARCH * @return {Object} IMAP command object */ function buildSEARCHCommand(query = {}, options = {}) { const command = { command: options.byUid ? 'UID SEARCH' : 'SEARCH' }; let isAscii = true; const buildTerm = query => { let list = []; Object.keys(query).forEach(key => { let params = []; const formatDate = date => date.toUTCString().replace(/^\w+, 0?(\d+) (\w+) (\d+).*/, '$1-$2-$3'); const escapeParam = param => { if (typeof param === 'number') { return { type: 'number', value: param }; } else if (typeof param === 'string') { if (/[\u0080-\uFFFF]/.test(param)) { isAscii = false; return { type: 'literal', value: (0, _common.fromTypedArray)((0, _emailjsMimeCodec.encode)(param)) // cast unicode string to pseudo-binary as imap-handler compiles strings as octets }; } return { type: 'string', value: param }; } else if (Object.prototype.toString.call(param) === '[object Date]') { // RFC 3501 allows for dates to be placed in // double-quotes or left without quotes. Some // servers (Yandex), do not like the double quotes, // so we treat the date as an atom. return { type: 'atom', value: formatDate(param) }; } else if (Array.isArray(param)) { return param.map(escapeParam); } else if (typeof param === 'object') { return buildTerm(param); } }; params.push({ type: 'atom', value: key.toUpperCase() }); [].concat(query[key] || []).forEach(param => { switch (key.toLowerCase()) { case 'uid': param = { type: 'sequence', value: param }; break; // The Gmail extension values of X-GM-THRID and // X-GM-MSGID are defined to be unsigned 64-bit integers // and they must not be quoted strings or the server // will report a parse error. case 'x-gm-thrid': case 'x-gm-msgid': param = { type: 'number', value: param }; break; default: param = escapeParam(param); } if (param) { params = params.concat(param || []); } }); list = list.concat(params || []); }); return list; }; command.attributes = buildTerm(query); // If any string input is using 8bit bytes, prepend the optional CHARSET argument if (!isAscii) { command.attributes.unshift({ type: 'atom', value: 'UTF-8' }); command.attributes.unshift({ type: 'atom', value: 'CHARSET' }); } return command; } /** * Creates an IMAP STORE command from the selected arguments */ function buildSTORECommand(sequence, action = '', flags = [], options = {}) { const command = { command: options.byUid ? 'UID STORE' : 'STORE', attributes: [{ type: 'sequence', value: sequence }] }; command.attributes.push({ type: 'atom', value: action.toUpperCase() + (options.silent ? '.SILENT' : '') }); command.attributes.push(flags.map(flag => { return { type: 'atom', value: flag }; })); return command; } //# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["../src/command-builder.js"],"names":["buildFETCHCommand","sequence","items","options","command","byUid","attributes","type","value","valueAsString","undefined","query","forEach","item","toUpperCase","trim","test","push","cmd","concat","e","length","pop","changedSince","buildXOAuth2Token","user","token","authData","join","buildSEARCHCommand","isAscii","buildTerm","list","Object","keys","key","params","formatDate","date","toUTCString","replace","escapeParam","param","prototype","toString","call","Array","isArray","map","toLowerCase","unshift","buildSTORECommand","action","flags","silent","flag"],"mappings":"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","sourcesContent":["import { parser } from 'emailjs-imap-handler'\nimport { encode } from 'emailjs-mime-codec'\nimport { encode as encodeBase64 } from 'emailjs-base64'\nimport {\n  fromTypedArray,\n  toTypedArray\n} from './common'\n\n/**\n * Builds a FETCH command\n *\n * @param {String} sequence Message range selector\n * @param {Array} items List of elements to fetch (eg. `['uid', 'envelope']`).\n * @param {Object} [options] Optional options object. Use `{byUid:true}` for `UID FETCH`\n * @returns {Object} Structured IMAP command\n */\nexport function buildFETCHCommand (sequence, items, options) {\n  const command = {\n    command: options.byUid ? 'UID FETCH' : 'FETCH',\n    attributes: [{\n      type: 'SEQUENCE',\n      value: sequence\n    }]\n  }\n\n  if (options.valueAsString !== undefined) {\n    command.valueAsString = options.valueAsString\n  }\n\n  let query = []\n\n  items.forEach((item) => {\n    item = item.toUpperCase().trim()\n\n    if (/^\\w+$/.test(item)) {\n      // alphanum strings can be used directly\n      query.push({\n        type: 'ATOM',\n        value: item\n      })\n    } else if (item) {\n      try {\n        // parse the value as a fake command, use only the attributes block\n        const cmd = parser(toTypedArray('* Z ' + item))\n        query = query.concat(cmd.attributes || [])\n      } catch (e) {\n        // if parse failed, use the original string as one entity\n        query.push({\n          type: 'ATOM',\n          value: item\n        })\n      }\n    }\n  })\n\n  if (query.length === 1) {\n    query = query.pop()\n  }\n\n  command.attributes.push(query)\n\n  if (options.changedSince) {\n    command.attributes.push([{\n      type: 'ATOM',\n      value: 'CHANGEDSINCE'\n    }, {\n      type: 'ATOM',\n      value: options.changedSince\n    }])\n  }\n\n  return command\n}\n\n/**\n * Builds a login token for XOAUTH2 authentication command\n *\n * @param {String} user E-mail address of the user\n * @param {String} token Valid access token for the user\n * @return {String} Base64 formatted login token\n */\nexport function buildXOAuth2Token (user = '', token) {\n  const authData = [\n    `user=${user}`,\n    `auth=Bearer ${token}`,\n    '',\n    ''\n  ]\n  return encodeBase64(authData.join('\\x01'))\n}\n\n/**\n * Compiles a search query into an IMAP command. Queries are composed as objects\n * where keys are search terms and values are term arguments. Only strings,\n * numbers and Dates are used. If the value is an array, the members of it\n * are processed separately (use this for terms that require multiple params).\n * If the value is a Date, it is converted to the form of \"01-Jan-1970\".\n * Subqueries (OR, NOT) are made up of objects\n *\n *    {unseen: true, header: [\"subject\", \"hello world\"]};\n *    SEARCH UNSEEN HEADER \"subject\" \"hello world\"\n *\n * @param {Object} query Search query\n * @param {Object} [options] Option object\n * @param {Boolean} [options.byUid] If ture, use UID SEARCH instead of SEARCH\n * @return {Object} IMAP command object\n */\nexport function buildSEARCHCommand (query = {}, options = {}) {\n  const command = {\n    command: options.byUid ? 'UID SEARCH' : 'SEARCH'\n  }\n\n  let isAscii = true\n\n  const buildTerm = (query) => {\n    let list = []\n\n    Object.keys(query).forEach((key) => {\n      let params = []\n      const formatDate = (date) => date.toUTCString().replace(/^\\w+, 0?(\\d+) (\\w+) (\\d+).*/, '$1-$2-$3')\n      const escapeParam = (param) => {\n        if (typeof param === 'number') {\n          return {\n            type: 'number',\n            value: param\n          }\n        } else if (typeof param === 'string') {\n          if (/[\\u0080-\\uFFFF]/.test(param)) {\n            isAscii = false\n            return {\n              type: 'literal',\n              value: fromTypedArray(encode(param)) // cast unicode string to pseudo-binary as imap-handler compiles strings as octets\n            }\n          }\n          return {\n            type: 'string',\n            value: param\n          }\n        } else if (Object.prototype.toString.call(param) === '[object Date]') {\n          // RFC 3501 allows for dates to be placed in\n          // double-quotes or left without quotes.  Some\n          // servers (Yandex), do not like the double quotes,\n          // so we treat the date as an atom.\n          return {\n            type: 'atom',\n            value: formatDate(param)\n          }\n        } else if (Array.isArray(param)) {\n          return param.map(escapeParam)\n        } else if (typeof param === 'object') {\n          return buildTerm(param)\n        }\n      }\n\n      params.push({\n        type: 'atom',\n        value: key.toUpperCase()\n      });\n\n      [].concat(query[key] || []).forEach((param) => {\n        switch (key.toLowerCase()) {\n          case 'uid':\n            param = {\n              type: 'sequence',\n              value: param\n            }\n            break\n          // The Gmail extension values of X-GM-THRID and\n          // X-GM-MSGID are defined to be unsigned 64-bit integers\n          // and they must not be quoted strings or the server\n          // will report a parse error.\n          case 'x-gm-thrid':\n          case 'x-gm-msgid':\n            param = {\n              type: 'number',\n              value: param\n            }\n            break\n          default:\n            param = escapeParam(param)\n        }\n        if (param) {\n          params = params.concat(param || [])\n        }\n      })\n      list = list.concat(params || [])\n    })\n\n    return list\n  }\n\n  command.attributes = buildTerm(query)\n\n  // If any string input is using 8bit bytes, prepend the optional CHARSET argument\n  if (!isAscii) {\n    command.attributes.unshift({\n      type: 'atom',\n      value: 'UTF-8'\n    })\n    command.attributes.unshift({\n      type: 'atom',\n      value: 'CHARSET'\n    })\n  }\n\n  return command\n}\n\n/**\n * Creates an IMAP STORE command from the selected arguments\n */\nexport function buildSTORECommand (sequence, action = '', flags = [], options = {}) {\n  const command = {\n    command: options.byUid ? 'UID STORE' : 'STORE',\n    attributes: [{\n      type: 'sequence',\n      value: sequence\n    }]\n  }\n\n  command.attributes.push({\n    type: 'atom',\n    value: action.toUpperCase() + (options.silent ? '.SILENT' : '')\n  })\n\n  command.attributes.push(flags.map((flag) => {\n    return {\n      type: 'atom',\n      value: flag\n    }\n  }))\n\n  return command\n}\n"]}