Tim van der Lippe | efb716a | 2020-12-01 12:54:04 +0000 | [diff] [blame] | 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 | } |