| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121 | 'use strict'const hexify = char => {  const h = char.charCodeAt(0).toString(16).toUpperCase()  return '0x' + (h.length % 2 ? '0' : '') + h}const parseError = (e, txt, context) => {  if (!txt) {    return {      message: e.message + ' while parsing empty string',      position: 0,    }  }  const badToken = e.message.match(/^Unexpected token (.) .*position\s+(\d+)/i)  const errIdx = badToken ? +badToken[2]    : e.message.match(/^Unexpected end of JSON.*/i) ? txt.length - 1    : null  const msg = badToken ? e.message.replace(/^Unexpected token ./, `Unexpected token ${      JSON.stringify(badToken[1])    } (${hexify(badToken[1])})`)    : e.message  if (errIdx !== null && errIdx !== undefined) {    const start = errIdx <= context ? 0      : errIdx - context    const end = errIdx + context >= txt.length ? txt.length      : errIdx + context    const slice = (start === 0 ? '' : '...') +      txt.slice(start, end) +      (end === txt.length ? '' : '...')    const near = txt === slice ? '' : 'near '    return {      message: msg + ` while parsing ${near}${JSON.stringify(slice)}`,      position: errIdx,    }  } else {    return {      message: msg + ` while parsing '${txt.slice(0, context * 2)}'`,      position: 0,    }  }}class JSONParseError extends SyntaxError {  constructor (er, txt, context, caller) {    context = context || 20    const metadata = parseError(er, txt, context)    super(metadata.message)    Object.assign(this, metadata)    this.code = 'EJSONPARSE'    this.systemError = er    Error.captureStackTrace(this, caller || this.constructor)  }  get name () { return this.constructor.name }  set name (n) {}  get [Symbol.toStringTag] () { return this.constructor.name }}const kIndent = Symbol.for('indent')const kNewline = Symbol.for('newline')// only respect indentation if we got a line break, otherwise squash it// things other than objects and arrays aren't indented, so ignore those// Important: in both of these regexps, the $1 capture group is the newline// or undefined, and the $2 capture group is the indent, or undefined.const formatRE = /^\s*[{\[]((?:\r?\n)+)([\s\t]*)/const emptyRE = /^(?:\{\}|\[\])((?:\r?\n)+)?$/const parseJson = (txt, reviver, context) => {  const parseText = stripBOM(txt)  context = context || 20  try {    // get the indentation so that we can save it back nicely    // if the file starts with {" then we have an indent of '', ie, none    // otherwise, pick the indentation of the next line after the first \n    // If the pattern doesn't match, then it means no indentation.    // JSON.stringify ignores symbols, so this is reasonably safe.    // if the string is '{}' or '[]', then use the default 2-space indent.    const [, newline = '\n', indent = '  '] = parseText.match(emptyRE) ||      parseText.match(formatRE) ||      [, '', '']    const result = JSON.parse(parseText, reviver)    if (result && typeof result === 'object') {      result[kNewline] = newline      result[kIndent] = indent    }    return result  } catch (e) {    if (typeof txt !== 'string' && !Buffer.isBuffer(txt)) {      const isEmptyArray = Array.isArray(txt) && txt.length === 0      throw Object.assign(new TypeError(        `Cannot parse ${isEmptyArray ? 'an empty array' : String(txt)}`      ), {        code: 'EJSONPARSE',        systemError: e,      })    }    throw new JSONParseError(e, parseText, context, parseJson)  }}// Remove byte order marker. This catches EF BB BF (the UTF-8 BOM)// because the buffer-to-string conversion in `fs.readFileSync()`// translates it to FEFF, the UTF-16 BOM.const stripBOM = txt => String(txt).replace(/^\uFEFF/, '')module.exports = parseJsonparseJson.JSONParseError = JSONParseErrorparseJson.noExceptions = (txt, reviver) => {  try {    return JSON.parse(stripBOM(txt), reviver)  } catch (e) {}}
 |