| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101 | "use strict";/** * Implementation of atob() according to the HTML and Infra specs, except that * instead of throwing INVALID_CHARACTER_ERR we return null. */function atob(data) {  if (arguments.length === 0) {    throw new TypeError("1 argument required, but only 0 present.");  }  // Web IDL requires DOMStrings to just be converted using ECMAScript  // ToString, which in our case amounts to using a template literal.  data = `${data}`;  // "Remove all ASCII whitespace from data."  data = data.replace(/[ \t\n\f\r]/g, "");  // "If data's length divides by 4 leaving no remainder, then: if data ends  // with one or two U+003D (=) code points, then remove them from data."  if (data.length % 4 === 0) {    data = data.replace(/==?$/, "");  }  // "If data's length divides by 4 leaving a remainder of 1, then return  // failure."  //  // "If data contains a code point that is not one of  //  // U+002B (+)  // U+002F (/)  // ASCII alphanumeric  //  // then return failure."  if (data.length % 4 === 1 || /[^+/0-9A-Za-z]/.test(data)) {    return null;  }  // "Let output be an empty byte sequence."  let output = "";  // "Let buffer be an empty buffer that can have bits appended to it."  //  // We append bits via left-shift and or.  accumulatedBits is used to track  // when we've gotten to 24 bits.  let buffer = 0;  let accumulatedBits = 0;  // "Let position be a position variable for data, initially pointing at the  // start of data."  //  // "While position does not point past the end of data:"  for (let i = 0; i < data.length; i++) {    // "Find the code point pointed to by position in the second column of    // Table 1: The Base 64 Alphabet of RFC 4648. Let n be the number given in    // the first cell of the same row.    //    // "Append to buffer the six bits corresponding to n, most significant bit    // first."    //    // atobLookup() implements the table from RFC 4648.    buffer <<= 6;    buffer |= atobLookup(data[i]);    accumulatedBits += 6;    // "If buffer has accumulated 24 bits, interpret them as three 8-bit    // big-endian numbers. Append three bytes with values equal to those    // numbers to output, in the same order, and then empty buffer."    if (accumulatedBits === 24) {      output += String.fromCharCode((buffer & 0xff0000) >> 16);      output += String.fromCharCode((buffer & 0xff00) >> 8);      output += String.fromCharCode(buffer & 0xff);      buffer = accumulatedBits = 0;    }    // "Advance position by 1."  }  // "If buffer is not empty, it contains either 12 or 18 bits. If it contains  // 12 bits, then discard the last four and interpret the remaining eight as  // an 8-bit big-endian number. If it contains 18 bits, then discard the last  // two and interpret the remaining 16 as two 8-bit big-endian numbers. Append  // the one or two bytes with values equal to those one or two numbers to  // output, in the same order."  if (accumulatedBits === 12) {    buffer >>= 4;    output += String.fromCharCode(buffer);  } else if (accumulatedBits === 18) {    buffer >>= 2;    output += String.fromCharCode((buffer & 0xff00) >> 8);    output += String.fromCharCode(buffer & 0xff);  }  // "Return output."  return output;}/** * A lookup table for atob(), which converts an ASCII character to the * corresponding six-bit number. */const keystr =  "ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz0123456789+/";function atobLookup(chr) {  const index = keystr.indexOf(chr);  // Throw exception if character is not in the lookup string; should not be hit in tests  return index < 0 ? undefined : index;}module.exports = atob;
 |