parse-json-string.js 1.6 KB

12345678910111213141516171819202122232425262728293031323334353637383940414243444546474849505152535455
  1. var uncurryThis = require('../internals/function-uncurry-this');
  2. var hasOwn = require('../internals/has-own-property');
  3. var $SyntaxError = SyntaxError;
  4. var $parseInt = parseInt;
  5. var fromCharCode = String.fromCharCode;
  6. var at = uncurryThis(''.charAt);
  7. var slice = uncurryThis(''.slice);
  8. var exec = uncurryThis(/./.exec);
  9. var codePoints = {
  10. '\\"': '"',
  11. '\\\\': '\\',
  12. '\\/': '/',
  13. '\\b': '\b',
  14. '\\f': '\f',
  15. '\\n': '\n',
  16. '\\r': '\r',
  17. '\\t': '\t'
  18. };
  19. var IS_4_HEX_DIGITS = /^[\da-f]{4}$/i;
  20. // eslint-disable-next-line regexp/no-control-character -- safe
  21. var IS_C0_CONTROL_CODE = /^[\u0000-\u001F]$/;
  22. module.exports = function (source, i) {
  23. var unterminated = true;
  24. var value = '';
  25. while (i < source.length) {
  26. var chr = at(source, i);
  27. if (chr == '\\') {
  28. var twoChars = slice(source, i, i + 2);
  29. if (hasOwn(codePoints, twoChars)) {
  30. value += codePoints[twoChars];
  31. i += 2;
  32. } else if (twoChars == '\\u') {
  33. i += 2;
  34. var fourHexDigits = slice(source, i, i + 4);
  35. if (!exec(IS_4_HEX_DIGITS, fourHexDigits)) throw $SyntaxError('Bad Unicode escape at: ' + i);
  36. value += fromCharCode($parseInt(fourHexDigits, 16));
  37. i += 4;
  38. } else throw $SyntaxError('Unknown escape sequence: "' + twoChars + '"');
  39. } else if (chr == '"') {
  40. unterminated = false;
  41. i++;
  42. break;
  43. } else {
  44. if (exec(IS_C0_CONTROL_CODE, chr)) throw $SyntaxError('Bad control character in string literal at: ' + i);
  45. value += chr;
  46. i++;
  47. }
  48. }
  49. if (unterminated) throw $SyntaxError('Unterminated string at: ' + i);
  50. return { value: value, end: i };
  51. };