'use strict' 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 } else { options = {} } } const protoAction = options.protoAction || 'error' const constructorAction = options.constructorAction || 'error' // BOM checker if (text && text.charCodeAt(0) === 0xFEFF) { text = text.slice(1) } // Parse normally, allowing exceptions const obj = JSON.parse(text, reviver) // options: 'error' (default) / 'remove' / 'ignore' if (protoAction === 'ignore' && constructorAction === 'ignore') { return obj } // Ignore null and non-objects if (obj === null || typeof obj !== 'object') { 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 scan(obj, { protoAction, constructorAction }) return obj } function scan (obj, { protoAction = 'error', constructorAction = 'error' } = {}) { 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 (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')) { // Avoid calling node.hasOwnProperty directly 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(node[key]) } } } } } function safeParse (text, reviver) { try { return parse(text, reviver) } catch (ignoreError) { return null } } module.exports = { parse, scan, safeParse }