| '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 = parseJson | 
| parseJson.JSONParseError = JSONParseError | 
|   | 
| parseJson.noExceptions = (txt, reviver) => { | 
|   try { | 
|     return JSON.parse(stripBOM(txt), reviver) | 
|   } catch (e) {} | 
| } |