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