index.js
  1  'use strict'
  2  
  3  const hexify = char => {
  4    const h = char.charCodeAt(0).toString(16).toUpperCase()
  5    return '0x' + (h.length % 2 ? '0' : '') + h
  6  }
  7  
  8  const parseError = (e, txt, context) => {
  9    if (!txt) {
 10      return {
 11        message: e.message + ' while parsing empty string',
 12        position: 0,
 13      }
 14    }
 15    const badToken = e.message.match(/^Unexpected token (.) .*position\s+(\d+)/i)
 16    const errIdx = badToken ? +badToken[2]
 17      : e.message.match(/^Unexpected end of JSON.*/i) ? txt.length - 1
 18      : null
 19  
 20    const msg = badToken ? e.message.replace(/^Unexpected token ./, `Unexpected token ${
 21        JSON.stringify(badToken[1])
 22      } (${hexify(badToken[1])})`)
 23      : e.message
 24  
 25    if (errIdx !== null && errIdx !== undefined) {
 26      const start = errIdx <= context ? 0
 27        : errIdx - context
 28  
 29      const end = errIdx + context >= txt.length ? txt.length
 30        : errIdx + context
 31  
 32      const slice = (start === 0 ? '' : '...') +
 33        txt.slice(start, end) +
 34        (end === txt.length ? '' : '...')
 35  
 36      const near = txt === slice ? '' : 'near '
 37  
 38      return {
 39        message: msg + ` while parsing ${near}${JSON.stringify(slice)}`,
 40        position: errIdx,
 41      }
 42    } else {
 43      return {
 44        message: msg + ` while parsing '${txt.slice(0, context * 2)}'`,
 45        position: 0,
 46      }
 47    }
 48  }
 49  
 50  class JSONParseError extends SyntaxError {
 51    constructor (er, txt, context, caller) {
 52      context = context || 20
 53      const metadata = parseError(er, txt, context)
 54      super(metadata.message)
 55      Object.assign(this, metadata)
 56      this.code = 'EJSONPARSE'
 57      this.systemError = er
 58      Error.captureStackTrace(this, caller || this.constructor)
 59    }
 60    get name () { return this.constructor.name }
 61    set name (n) {}
 62    get [Symbol.toStringTag] () { return this.constructor.name }
 63  }
 64  
 65  const kIndent = Symbol.for('indent')
 66  const kNewline = Symbol.for('newline')
 67  // only respect indentation if we got a line break, otherwise squash it
 68  // things other than objects and arrays aren't indented, so ignore those
 69  // Important: in both of these regexps, the $1 capture group is the newline
 70  // or undefined, and the $2 capture group is the indent, or undefined.
 71  const formatRE = /^\s*[{\[]((?:\r?\n)+)([\s\t]*)/
 72  const emptyRE = /^(?:\{\}|\[\])((?:\r?\n)+)?$/
 73  
 74  const parseJson = (txt, reviver, context) => {
 75    const parseText = stripBOM(txt)
 76    context = context || 20
 77    try {
 78      // get the indentation so that we can save it back nicely
 79      // if the file starts with {" then we have an indent of '', ie, none
 80      // otherwise, pick the indentation of the next line after the first \n
 81      // If the pattern doesn't match, then it means no indentation.
 82      // JSON.stringify ignores symbols, so this is reasonably safe.
 83      // if the string is '{}' or '[]', then use the default 2-space indent.
 84      const [, newline = '\n', indent = '  '] = parseText.match(emptyRE) ||
 85        parseText.match(formatRE) ||
 86        [, '', '']
 87  
 88      const result = JSON.parse(parseText, reviver)
 89      if (result && typeof result === 'object') {
 90        result[kNewline] = newline
 91        result[kIndent] = indent
 92      }
 93      return result
 94    } catch (e) {
 95      if (typeof txt !== 'string' && !Buffer.isBuffer(txt)) {
 96        const isEmptyArray = Array.isArray(txt) && txt.length === 0
 97        throw Object.assign(new TypeError(
 98          `Cannot parse ${isEmptyArray ? 'an empty array' : String(txt)}`
 99        ), {
100          code: 'EJSONPARSE',
101          systemError: e,
102        })
103      }
104  
105      throw new JSONParseError(e, parseText, context, parseJson)
106    }
107  }
108  
109  // Remove byte order marker. This catches EF BB BF (the UTF-8 BOM)
110  // because the buffer-to-string conversion in `fs.readFileSync()`
111  // translates it to FEFF, the UTF-16 BOM.
112  const stripBOM = txt => String(txt).replace(/^\uFEFF/, '')
113  
114  module.exports = parseJson
115  parseJson.JSONParseError = JSONParseError
116  
117  parseJson.noExceptions = (txt, reviver) => {
118    try {
119      return JSON.parse(stripBOM(txt), reviver)
120    } catch (e) {}
121  }