blob: 7a61cfc2d24dceb3530988884f70f51f398d5855 [file] [log] [blame]
Olivier Deprezf4ef2d02021-04-20 13:36:24 +02001"""JSON token scanner
2"""
3import re
4try:
5 from _json import make_scanner as c_make_scanner
6except ImportError:
7 c_make_scanner = None
8
9__all__ = ['make_scanner']
10
11NUMBER_RE = re.compile(
12 r'(-?(?:0|[1-9]\d*))(\.\d+)?([eE][-+]?\d+)?',
13 (re.VERBOSE | re.MULTILINE | re.DOTALL))
14
15def py_make_scanner(context):
16 parse_object = context.parse_object
17 parse_array = context.parse_array
18 parse_string = context.parse_string
19 match_number = NUMBER_RE.match
20 strict = context.strict
21 parse_float = context.parse_float
22 parse_int = context.parse_int
23 parse_constant = context.parse_constant
24 object_hook = context.object_hook
25 object_pairs_hook = context.object_pairs_hook
26 memo = context.memo
27
28 def _scan_once(string, idx):
29 try:
30 nextchar = string[idx]
31 except IndexError:
32 raise StopIteration(idx) from None
33
34 if nextchar == '"':
35 return parse_string(string, idx + 1, strict)
36 elif nextchar == '{':
37 return parse_object((string, idx + 1), strict,
38 _scan_once, object_hook, object_pairs_hook, memo)
39 elif nextchar == '[':
40 return parse_array((string, idx + 1), _scan_once)
41 elif nextchar == 'n' and string[idx:idx + 4] == 'null':
42 return None, idx + 4
43 elif nextchar == 't' and string[idx:idx + 4] == 'true':
44 return True, idx + 4
45 elif nextchar == 'f' and string[idx:idx + 5] == 'false':
46 return False, idx + 5
47
48 m = match_number(string, idx)
49 if m is not None:
50 integer, frac, exp = m.groups()
51 if frac or exp:
52 res = parse_float(integer + (frac or '') + (exp or ''))
53 else:
54 res = parse_int(integer)
55 return res, m.end()
56 elif nextchar == 'N' and string[idx:idx + 3] == 'NaN':
57 return parse_constant('NaN'), idx + 3
58 elif nextchar == 'I' and string[idx:idx + 8] == 'Infinity':
59 return parse_constant('Infinity'), idx + 8
60 elif nextchar == '-' and string[idx:idx + 9] == '-Infinity':
61 return parse_constant('-Infinity'), idx + 9
62 else:
63 raise StopIteration(idx)
64
65 def scan_once(string, idx):
66 try:
67 return _scan_once(string, idx)
68 finally:
69 memo.clear()
70
71 return scan_once
72
73make_scanner = c_make_scanner or py_make_scanner