You can not select more than 25 topics
			Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
		
		
		
		
		
			
		
			
				
					113 lines
				
				3.3 KiB
			
		
		
			
		
	
	
					113 lines
				
				3.3 KiB
			| 
								 
											3 years ago
										 
									 | 
							
								'use strict'
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								const hasBuffer = typeof Buffer !== 'undefined'
							 | 
						||
| 
								 | 
							
								const suspectProtoRx = /"(?:_|\\u005[Ff])(?:_|\\u005[Ff])(?:p|\\u0070)(?:r|\\u0072)(?:o|\\u006[Ff])(?:t|\\u0074)(?:o|\\u006[Ff])(?:_|\\u005[Ff])(?:_|\\u005[Ff])"\s*:/
							 | 
						||
| 
								 | 
							
								const suspectConstructorRx = /"(?:c|\\u0063)(?:o|\\u006[Ff])(?:n|\\u006[Ee])(?:s|\\u0073)(?:t|\\u0074)(?:r|\\u0072)(?:u|\\u0075)(?:c|\\u0063)(?:t|\\u0074)(?:o|\\u006[Ff])(?:r|\\u0072)"\s*:/
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								function parse (text, reviver, options) {
							 | 
						||
| 
								 | 
							
								  // Normalize arguments
							 | 
						||
| 
								 | 
							
								  if (options == null) {
							 | 
						||
| 
								 | 
							
								    if (reviver !== null && typeof reviver === 'object') {
							 | 
						||
| 
								 | 
							
								      options = reviver
							 | 
						||
| 
								 | 
							
								      reviver = undefined
							 | 
						||
| 
								 | 
							
								    }
							 | 
						||
| 
								 | 
							
								  }
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								  if (hasBuffer && Buffer.isBuffer(text)) {
							 | 
						||
| 
								 | 
							
								    text = text.toString()
							 | 
						||
| 
								 | 
							
								  }
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								  // BOM checker
							 | 
						||
| 
								 | 
							
								  if (text && text.charCodeAt(0) === 0xFEFF) {
							 | 
						||
| 
								 | 
							
								    text = text.slice(1)
							 | 
						||
| 
								 | 
							
								  }
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								  // Parse normally, allowing exceptions
							 | 
						||
| 
								 | 
							
								  const obj = JSON.parse(text, reviver)
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								  // Ignore null and non-objects
							 | 
						||
| 
								 | 
							
								  if (obj === null || typeof obj !== 'object') {
							 | 
						||
| 
								 | 
							
								    return obj
							 | 
						||
| 
								 | 
							
								  }
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								  const protoAction = (options && options.protoAction) || 'error'
							 | 
						||
| 
								 | 
							
								  const constructorAction = (options && options.constructorAction) || 'error'
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								  // options: 'error' (default) / 'remove' / 'ignore'
							 | 
						||
| 
								 | 
							
								  if (protoAction === 'ignore' && constructorAction === 'ignore') {
							 | 
						||
| 
								 | 
							
								    return obj
							 | 
						||
| 
								 | 
							
								  }
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								  if (protoAction !== 'ignore' && constructorAction !== 'ignore') {
							 | 
						||
| 
								 | 
							
								    if (suspectProtoRx.test(text) === false && suspectConstructorRx.test(text) === false) {
							 | 
						||
| 
								 | 
							
								      return obj
							 | 
						||
| 
								 | 
							
								    }
							 | 
						||
| 
								 | 
							
								  } else if (protoAction !== 'ignore' && constructorAction === 'ignore') {
							 | 
						||
| 
								 | 
							
								    if (suspectProtoRx.test(text) === false) {
							 | 
						||
| 
								 | 
							
								      return obj
							 | 
						||
| 
								 | 
							
								    }
							 | 
						||
| 
								 | 
							
								  } else {
							 | 
						||
| 
								 | 
							
								    if (suspectConstructorRx.test(text) === false) {
							 | 
						||
| 
								 | 
							
								      return obj
							 | 
						||
| 
								 | 
							
								    }
							 | 
						||
| 
								 | 
							
								  }
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								  // Scan result for proto keys
							 | 
						||
| 
								 | 
							
								  return filter(obj, { protoAction, constructorAction, safe: options && options.safe })
							 | 
						||
| 
								 | 
							
								}
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								function filter (obj, { protoAction = 'error', constructorAction = 'error', safe } = {}) {
							 | 
						||
| 
								 | 
							
								  let next = [obj]
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								  while (next.length) {
							 | 
						||
| 
								 | 
							
								    const nodes = next
							 | 
						||
| 
								 | 
							
								    next = []
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								    for (const node of nodes) {
							 | 
						||
| 
								 | 
							
								      if (protoAction !== 'ignore' && Object.prototype.hasOwnProperty.call(node, '__proto__')) { // Avoid calling node.hasOwnProperty directly
							 | 
						||
| 
								 | 
							
								        if (safe === true) {
							 | 
						||
| 
								 | 
							
								          return null
							 | 
						||
| 
								 | 
							
								        } else if (protoAction === 'error') {
							 | 
						||
| 
								 | 
							
								          throw new SyntaxError('Object contains forbidden prototype property')
							 | 
						||
| 
								 | 
							
								        }
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								        delete node.__proto__ // eslint-disable-line no-proto
							 | 
						||
| 
								 | 
							
								      }
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								      if (constructorAction !== 'ignore' &&
							 | 
						||
| 
								 | 
							
								          Object.prototype.hasOwnProperty.call(node, 'constructor') &&
							 | 
						||
| 
								 | 
							
								          Object.prototype.hasOwnProperty.call(node.constructor, 'prototype')) { // Avoid calling node.hasOwnProperty directly
							 | 
						||
| 
								 | 
							
								        if (safe === true) {
							 | 
						||
| 
								 | 
							
								          return null
							 | 
						||
| 
								 | 
							
								        } else if (constructorAction === 'error') {
							 | 
						||
| 
								 | 
							
								          throw new SyntaxError('Object contains forbidden prototype property')
							 | 
						||
| 
								 | 
							
								        }
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								        delete node.constructor
							 | 
						||
| 
								 | 
							
								      }
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								      for (const key in node) {
							 | 
						||
| 
								 | 
							
								        const value = node[key]
							 | 
						||
| 
								 | 
							
								        if (value && typeof value === 'object') {
							 | 
						||
| 
								 | 
							
								          next.push(value)
							 | 
						||
| 
								 | 
							
								        }
							 | 
						||
| 
								 | 
							
								      }
							 | 
						||
| 
								 | 
							
								    }
							 | 
						||
| 
								 | 
							
								  }
							 | 
						||
| 
								 | 
							
								  return obj
							 | 
						||
| 
								 | 
							
								}
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								function safeParse (text, reviver) {
							 | 
						||
| 
								 | 
							
								  try {
							 | 
						||
| 
								 | 
							
								    return parse(text, reviver, { safe: true })
							 | 
						||
| 
								 | 
							
								  } catch (ignoreError) {
							 | 
						||
| 
								 | 
							
								    return null
							 | 
						||
| 
								 | 
							
								  }
							 | 
						||
| 
								 | 
							
								}
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								module.exports = parse
							 | 
						||
| 
								 | 
							
								module.exports.default = parse
							 | 
						||
| 
								 | 
							
								module.exports.parse = parse
							 | 
						||
| 
								 | 
							
								module.exports.safeParse = safeParse
							 | 
						||
| 
								 | 
							
								module.exports.scan = filter
							 |