Blame view

天文台pc/tianwentai-ui/node_modules/json-parse-even-better-errors/index.js 3.81 KB
bc518174   王天杨   提交两个项目文件
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
  '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) {}
  }