74 lines
2.4 KiB
Python
74 lines
2.4 KiB
Python
|
"""JSON token scanner
|
||
|
"""
|
||
|
import re
|
||
|
try:
|
||
|
from _json import make_scanner as c_make_scanner
|
||
|
except ImportError:
|
||
|
c_make_scanner = None
|
||
|
|
||
|
__all__ = ['make_scanner']
|
||
|
|
||
|
NUMBER_RE = re.compile(
|
||
|
r'(-?(?:0|[1-9]\d*))(\.\d+)?([eE][-+]?\d+)?',
|
||
|
(re.VERBOSE | re.MULTILINE | re.DOTALL))
|
||
|
|
||
|
def py_make_scanner(context):
|
||
|
parse_object = context.parse_object
|
||
|
parse_array = context.parse_array
|
||
|
parse_string = context.parse_string
|
||
|
match_number = NUMBER_RE.match
|
||
|
strict = context.strict
|
||
|
parse_float = context.parse_float
|
||
|
parse_int = context.parse_int
|
||
|
parse_constant = context.parse_constant
|
||
|
object_hook = context.object_hook
|
||
|
object_pairs_hook = context.object_pairs_hook
|
||
|
memo = context.memo
|
||
|
|
||
|
def _scan_once(string, idx):
|
||
|
try:
|
||
|
nextchar = string[idx]
|
||
|
except IndexError:
|
||
|
raise StopIteration(idx)
|
||
|
|
||
|
if nextchar == '"':
|
||
|
return parse_string(string, idx + 1, strict)
|
||
|
elif nextchar == '{':
|
||
|
return parse_object((string, idx + 1), strict,
|
||
|
_scan_once, object_hook, object_pairs_hook, memo)
|
||
|
elif nextchar == '[':
|
||
|
return parse_array((string, idx + 1), _scan_once)
|
||
|
elif nextchar == 'n' and string[idx:idx + 4] == 'null':
|
||
|
return None, idx + 4
|
||
|
elif nextchar == 't' and string[idx:idx + 4] == 'true':
|
||
|
return True, idx + 4
|
||
|
elif nextchar == 'f' and string[idx:idx + 5] == 'false':
|
||
|
return False, idx + 5
|
||
|
|
||
|
m = match_number(string, idx)
|
||
|
if m is not None:
|
||
|
integer, frac, exp = m.groups()
|
||
|
if frac or exp:
|
||
|
res = parse_float(integer + (frac or '') + (exp or ''))
|
||
|
else:
|
||
|
res = parse_int(integer)
|
||
|
return res, m.end()
|
||
|
elif nextchar == 'N' and string[idx:idx + 3] == 'NaN':
|
||
|
return parse_constant('NaN'), idx + 3
|
||
|
elif nextchar == 'I' and string[idx:idx + 8] == 'Infinity':
|
||
|
return parse_constant('Infinity'), idx + 8
|
||
|
elif nextchar == '-' and string[idx:idx + 9] == '-Infinity':
|
||
|
return parse_constant('-Infinity'), idx + 9
|
||
|
else:
|
||
|
raise StopIteration(idx)
|
||
|
|
||
|
def scan_once(string, idx):
|
||
|
try:
|
||
|
return _scan_once(string, idx)
|
||
|
finally:
|
||
|
memo.clear()
|
||
|
|
||
|
return _scan_once
|
||
|
|
||
|
make_scanner = c_make_scanner or py_make_scanner
|