| 
									
										
										
										
											2008-05-08 14:29:10 +00:00
										 |  |  | """Implementation of JSONDecoder
 | 
					
						
							|  |  |  | """
 | 
					
						
							| 
									
										
										
										
											2009-05-02 12:36:44 +00:00
										 |  |  | import binascii | 
					
						
							| 
									
										
										
										
											2008-05-08 14:29:10 +00:00
										 |  |  | import re | 
					
						
							|  |  |  | import sys | 
					
						
							| 
									
										
										
										
											2009-05-02 12:36:44 +00:00
										 |  |  | import struct | 
					
						
							| 
									
										
										
										
											2008-05-08 14:29:10 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-05-14 06:47:51 +03:00
										 |  |  | from json import scanner | 
					
						
							| 
									
										
										
										
											2008-05-08 14:29:10 +00:00
										 |  |  | try: | 
					
						
							|  |  |  |     from _json import scanstring as c_scanstring | 
					
						
							|  |  |  | except ImportError: | 
					
						
							|  |  |  |     c_scanstring = None | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | __all__ = ['JSONDecoder'] | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | FLAGS = re.VERBOSE | re.MULTILINE | re.DOTALL | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-05-02 12:36:44 +00:00
										 |  |  | def _floatconstants(): | 
					
						
							|  |  |  |     _BYTES = binascii.unhexlify(b'7FF80000000000007FF0000000000000') | 
					
						
							|  |  |  |     if sys.byteorder != 'big': | 
					
						
							|  |  |  |         _BYTES = _BYTES[:8][::-1] + _BYTES[8:][::-1] | 
					
						
							|  |  |  |     nan, inf = struct.unpack('dd', _BYTES) | 
					
						
							|  |  |  |     return nan, inf, -inf | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | NaN, PosInf, NegInf = _floatconstants() | 
					
						
							| 
									
										
										
										
											2008-05-08 14:29:10 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | def linecol(doc, pos): | 
					
						
							| 
									
										
										
										
											2008-10-16 21:17:24 +00:00
										 |  |  |     if isinstance(doc, bytes): | 
					
						
							|  |  |  |         newline = b'\n' | 
					
						
							|  |  |  |     else: | 
					
						
							|  |  |  |         newline = '\n' | 
					
						
							|  |  |  |     lineno = doc.count(newline, 0, pos) + 1 | 
					
						
							| 
									
										
										
										
											2008-05-08 14:29:10 +00:00
										 |  |  |     if lineno == 1: | 
					
						
							|  |  |  |         colno = pos | 
					
						
							|  |  |  |     else: | 
					
						
							| 
									
										
										
										
											2008-10-16 21:17:24 +00:00
										 |  |  |         colno = pos - doc.rindex(newline, 0, pos) | 
					
						
							| 
									
										
										
										
											2008-05-08 14:29:10 +00:00
										 |  |  |     return lineno, colno | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | def errmsg(msg, doc, pos, end=None): | 
					
						
							| 
									
										
										
										
											2009-05-02 12:36:44 +00:00
										 |  |  |     # Note that this function is called from _json | 
					
						
							| 
									
										
										
										
											2008-05-08 14:29:10 +00:00
										 |  |  |     lineno, colno = linecol(doc, pos) | 
					
						
							|  |  |  |     if end is None: | 
					
						
							|  |  |  |         fmt = '{0}: line {1} column {2} (char {3})' | 
					
						
							|  |  |  |         return fmt.format(msg, lineno, colno, pos) | 
					
						
							| 
									
										
										
										
											2009-05-02 12:36:44 +00:00
										 |  |  |         #fmt = '%s: line %d column %d (char %d)' | 
					
						
							|  |  |  |         #return fmt % (msg, lineno, colno, pos) | 
					
						
							| 
									
										
										
										
											2008-05-08 14:29:10 +00:00
										 |  |  |     endlineno, endcolno = linecol(doc, end) | 
					
						
							|  |  |  |     fmt = '{0}: line {1} column {2} - line {3} column {4} (char {5} - {6})' | 
					
						
							|  |  |  |     return fmt.format(msg, lineno, colno, endlineno, endcolno, pos, end) | 
					
						
							| 
									
										
										
										
											2009-05-02 12:36:44 +00:00
										 |  |  |     #fmt = '%s: line %d column %d - line %d column %d (char %d - %d)' | 
					
						
							|  |  |  |     #return fmt % (msg, lineno, colno, endlineno, endcolno, pos, end) | 
					
						
							| 
									
										
										
										
											2008-05-08 14:29:10 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | _CONSTANTS = { | 
					
						
							|  |  |  |     '-Infinity': NegInf, | 
					
						
							|  |  |  |     'Infinity': PosInf, | 
					
						
							|  |  |  |     'NaN': NaN, | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | STRINGCHUNK = re.compile(r'(.*?)(["\\\x00-\x1f])', FLAGS) | 
					
						
							|  |  |  | BACKSLASH = { | 
					
						
							|  |  |  |     '"': '"', '\\': '\\', '/': '/', | 
					
						
							|  |  |  |     'b': '\b', 'f': '\f', 'n': '\n', 'r': '\r', 't': '\t', | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-05-02 12:36:44 +00:00
										 |  |  | def py_scanstring(s, end, strict=True, | 
					
						
							|  |  |  |         _b=BACKSLASH, _m=STRINGCHUNK.match): | 
					
						
							|  |  |  |     """Scan the string s for a JSON string. End is the index of the
 | 
					
						
							|  |  |  |     character in s after the quote that started the JSON string. | 
					
						
							|  |  |  |     Unescapes all valid JSON string escape sequences and raises ValueError | 
					
						
							|  |  |  |     on attempt to decode an invalid string. If strict is False then literal | 
					
						
							|  |  |  |     control characters are allowed in the string. | 
					
						
							| 
									
										
										
										
											2008-05-08 14:29:10 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-05-02 12:36:44 +00:00
										 |  |  |     Returns a tuple of the decoded string and the index of the character in s | 
					
						
							|  |  |  |     after the end quote."""
 | 
					
						
							| 
									
										
										
										
											2008-05-08 14:29:10 +00:00
										 |  |  |     chunks = [] | 
					
						
							|  |  |  |     _append = chunks.append | 
					
						
							|  |  |  |     begin = end - 1 | 
					
						
							|  |  |  |     while 1: | 
					
						
							|  |  |  |         chunk = _m(s, end) | 
					
						
							|  |  |  |         if chunk is None: | 
					
						
							|  |  |  |             raise ValueError( | 
					
						
							|  |  |  |                 errmsg("Unterminated string starting at", s, begin)) | 
					
						
							|  |  |  |         end = chunk.end() | 
					
						
							|  |  |  |         content, terminator = chunk.groups() | 
					
						
							| 
									
										
										
										
											2009-05-02 12:36:44 +00:00
										 |  |  |         # Content is contains zero or more unescaped string characters | 
					
						
							| 
									
										
										
										
											2008-05-08 14:29:10 +00:00
										 |  |  |         if content: | 
					
						
							|  |  |  |             _append(content) | 
					
						
							| 
									
										
										
										
											2009-05-02 12:36:44 +00:00
										 |  |  |         # Terminator is the end of string, a literal control character, | 
					
						
							|  |  |  |         # or a backslash denoting that an escape sequence follows | 
					
						
							| 
									
										
										
										
											2008-05-08 14:29:10 +00:00
										 |  |  |         if terminator == '"': | 
					
						
							|  |  |  |             break | 
					
						
							|  |  |  |         elif terminator != '\\': | 
					
						
							|  |  |  |             if strict: | 
					
						
							| 
									
										
										
										
											2009-05-02 12:36:44 +00:00
										 |  |  |                 #msg = "Invalid control character %r at" % (terminator,) | 
					
						
							| 
									
										
										
										
											2008-05-08 14:29:10 +00:00
										 |  |  |                 msg = "Invalid control character {0!r} at".format(terminator) | 
					
						
							|  |  |  |                 raise ValueError(errmsg(msg, s, end)) | 
					
						
							|  |  |  |             else: | 
					
						
							|  |  |  |                 _append(terminator) | 
					
						
							|  |  |  |                 continue | 
					
						
							|  |  |  |         try: | 
					
						
							|  |  |  |             esc = s[end] | 
					
						
							|  |  |  |         except IndexError: | 
					
						
							|  |  |  |             raise ValueError( | 
					
						
							|  |  |  |                 errmsg("Unterminated string starting at", s, begin)) | 
					
						
							| 
									
										
										
										
											2009-05-02 12:36:44 +00:00
										 |  |  |         # If not a unicode escape sequence, must be in the lookup table | 
					
						
							| 
									
										
										
										
											2008-05-08 14:29:10 +00:00
										 |  |  |         if esc != 'u': | 
					
						
							|  |  |  |             try: | 
					
						
							| 
									
										
										
										
											2009-05-02 12:36:44 +00:00
										 |  |  |                 char = _b[esc] | 
					
						
							| 
									
										
										
										
											2008-05-08 14:29:10 +00:00
										 |  |  |             except KeyError: | 
					
						
							|  |  |  |                 msg = "Invalid \\escape: {0!r}".format(esc) | 
					
						
							|  |  |  |                 raise ValueError(errmsg(msg, s, end)) | 
					
						
							|  |  |  |             end += 1 | 
					
						
							|  |  |  |         else: | 
					
						
							|  |  |  |             esc = s[end + 1:end + 5] | 
					
						
							|  |  |  |             next_end = end + 5 | 
					
						
							| 
									
										
										
										
											2009-05-02 12:36:44 +00:00
										 |  |  |             if len(esc) != 4: | 
					
						
							|  |  |  |                 msg = "Invalid \\uXXXX escape" | 
					
						
							| 
									
										
										
										
											2008-05-08 14:29:10 +00:00
										 |  |  |                 raise ValueError(errmsg(msg, s, end)) | 
					
						
							| 
									
										
										
										
											2009-05-02 12:36:44 +00:00
										 |  |  |             uni = int(esc, 16) | 
					
						
							| 
									
										
										
										
											2011-09-28 07:41:54 +02:00
										 |  |  |             if 0xd800 <= uni <= 0xdbff: | 
					
						
							| 
									
										
										
										
											2009-05-02 12:36:44 +00:00
										 |  |  |                 msg = "Invalid \\uXXXX\\uXXXX surrogate pair" | 
					
						
							|  |  |  |                 if not s[end + 5:end + 7] == '\\u': | 
					
						
							|  |  |  |                     raise ValueError(errmsg(msg, s, end)) | 
					
						
							|  |  |  |                 esc2 = s[end + 7:end + 11] | 
					
						
							|  |  |  |                 if len(esc2) != 4: | 
					
						
							|  |  |  |                     raise ValueError(errmsg(msg, s, end)) | 
					
						
							|  |  |  |                 uni2 = int(esc2, 16) | 
					
						
							|  |  |  |                 uni = 0x10000 + (((uni - 0xd800) << 10) | (uni2 - 0xdc00)) | 
					
						
							|  |  |  |                 next_end += 6 | 
					
						
							|  |  |  |             char = chr(uni) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-05-08 14:29:10 +00:00
										 |  |  |             end = next_end | 
					
						
							| 
									
										
										
										
											2009-05-02 12:36:44 +00:00
										 |  |  |         _append(char) | 
					
						
							| 
									
										
										
										
											2008-05-08 14:29:10 +00:00
										 |  |  |     return ''.join(chunks), end | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-05-02 12:36:44 +00:00
										 |  |  | # Use speedup if available | 
					
						
							|  |  |  | scanstring = c_scanstring or py_scanstring | 
					
						
							| 
									
										
										
										
											2008-05-08 14:29:10 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-05-02 12:36:44 +00:00
										 |  |  | WHITESPACE = re.compile(r'[ \t\n\r]*', FLAGS) | 
					
						
							|  |  |  | WHITESPACE_STR = ' \t\n\r' | 
					
						
							| 
									
										
										
										
											2008-05-08 14:29:10 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-05-02 12:36:44 +00:00
										 |  |  | def JSONObject(s_and_end, strict, scan_once, object_hook, object_pairs_hook, | 
					
						
							| 
									
										
										
										
											2010-09-04 20:16:53 +00:00
										 |  |  |                memo=None, _w=WHITESPACE.match, _ws=WHITESPACE_STR): | 
					
						
							| 
									
										
										
										
											2009-05-02 12:36:44 +00:00
										 |  |  |     s, end = s_and_end | 
					
						
							| 
									
										
										
										
											2009-04-21 03:09:17 +00:00
										 |  |  |     pairs = [] | 
					
						
							|  |  |  |     pairs_append = pairs.append | 
					
						
							| 
									
										
										
										
											2010-09-04 20:16:53 +00:00
										 |  |  |     # Backwards compatibility | 
					
						
							|  |  |  |     if memo is None: | 
					
						
							|  |  |  |         memo = {} | 
					
						
							|  |  |  |     memo_get = memo.setdefault | 
					
						
							| 
									
										
										
										
											2009-05-02 12:36:44 +00:00
										 |  |  |     # Use a slice to prevent IndexError from being raised, the following | 
					
						
							|  |  |  |     # check will raise a more specific ValueError if the string is empty | 
					
						
							| 
									
										
										
										
											2008-05-08 14:29:10 +00:00
										 |  |  |     nextchar = s[end:end + 1] | 
					
						
							| 
									
										
										
										
											2009-05-02 12:36:44 +00:00
										 |  |  |     # Normally we expect nextchar == '"' | 
					
						
							| 
									
										
										
										
											2008-05-08 14:29:10 +00:00
										 |  |  |     if nextchar != '"': | 
					
						
							| 
									
										
										
										
											2009-05-02 12:36:44 +00:00
										 |  |  |         if nextchar in _ws: | 
					
						
							|  |  |  |             end = _w(s, end).end() | 
					
						
							|  |  |  |             nextchar = s[end:end + 1] | 
					
						
							|  |  |  |         # Trivial empty object | 
					
						
							|  |  |  |         if nextchar == '}': | 
					
						
							| 
									
										
										
										
											2011-04-13 07:10:13 +03:00
										 |  |  |             if object_pairs_hook is not None: | 
					
						
							|  |  |  |                 result = object_pairs_hook(pairs) | 
					
						
							|  |  |  |                 return result, end | 
					
						
							|  |  |  |             pairs = {} | 
					
						
							|  |  |  |             if object_hook is not None: | 
					
						
							|  |  |  |                 pairs = object_hook(pairs) | 
					
						
							| 
									
										
										
										
											2009-05-02 12:36:44 +00:00
										 |  |  |             return pairs, end + 1 | 
					
						
							|  |  |  |         elif nextchar != '"': | 
					
						
							|  |  |  |             raise ValueError(errmsg("Expecting property name", s, end)) | 
					
						
							| 
									
										
										
										
											2008-05-08 14:29:10 +00:00
										 |  |  |     end += 1 | 
					
						
							|  |  |  |     while True: | 
					
						
							| 
									
										
										
										
											2009-05-02 12:36:44 +00:00
										 |  |  |         key, end = scanstring(s, end, strict) | 
					
						
							| 
									
										
										
										
											2010-09-04 20:16:53 +00:00
										 |  |  |         key = memo_get(key, key) | 
					
						
							| 
									
										
										
										
											2009-05-02 12:36:44 +00:00
										 |  |  |         # To skip some function call overhead we optimize the fast paths where | 
					
						
							|  |  |  |         # the JSON key separator is ": " or just ":". | 
					
						
							| 
									
										
										
										
											2008-05-08 14:29:10 +00:00
										 |  |  |         if s[end:end + 1] != ':': | 
					
						
							| 
									
										
										
										
											2009-05-02 12:36:44 +00:00
										 |  |  |             end = _w(s, end).end() | 
					
						
							|  |  |  |             if s[end:end + 1] != ':': | 
					
						
							|  |  |  |                 raise ValueError(errmsg("Expecting : delimiter", s, end)) | 
					
						
							|  |  |  |         end += 1 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-05-08 14:29:10 +00:00
										 |  |  |         try: | 
					
						
							| 
									
										
										
										
											2009-05-02 12:36:44 +00:00
										 |  |  |             if s[end] in _ws: | 
					
						
							|  |  |  |                 end += 1 | 
					
						
							|  |  |  |                 if s[end] in _ws: | 
					
						
							|  |  |  |                     end = _w(s, end + 1).end() | 
					
						
							|  |  |  |         except IndexError: | 
					
						
							|  |  |  |             pass | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         try: | 
					
						
							|  |  |  |             value, end = scan_once(s, end) | 
					
						
							| 
									
										
										
										
											2008-05-08 14:29:10 +00:00
										 |  |  |         except StopIteration: | 
					
						
							|  |  |  |             raise ValueError(errmsg("Expecting object", s, end)) | 
					
						
							| 
									
										
										
										
											2009-04-21 03:09:17 +00:00
										 |  |  |         pairs_append((key, value)) | 
					
						
							| 
									
										
										
										
											2009-05-02 12:36:44 +00:00
										 |  |  |         try: | 
					
						
							|  |  |  |             nextchar = s[end] | 
					
						
							|  |  |  |             if nextchar in _ws: | 
					
						
							|  |  |  |                 end = _w(s, end + 1).end() | 
					
						
							|  |  |  |                 nextchar = s[end] | 
					
						
							|  |  |  |         except IndexError: | 
					
						
							|  |  |  |             nextchar = '' | 
					
						
							| 
									
										
										
										
											2008-05-08 14:29:10 +00:00
										 |  |  |         end += 1 | 
					
						
							| 
									
										
										
										
											2009-05-02 12:36:44 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-05-08 14:29:10 +00:00
										 |  |  |         if nextchar == '}': | 
					
						
							|  |  |  |             break | 
					
						
							| 
									
										
										
										
											2009-05-02 12:36:44 +00:00
										 |  |  |         elif nextchar != ',': | 
					
						
							| 
									
										
										
										
											2008-05-08 14:29:10 +00:00
										 |  |  |             raise ValueError(errmsg("Expecting , delimiter", s, end - 1)) | 
					
						
							|  |  |  |         end = _w(s, end).end() | 
					
						
							|  |  |  |         nextchar = s[end:end + 1] | 
					
						
							|  |  |  |         end += 1 | 
					
						
							|  |  |  |         if nextchar != '"': | 
					
						
							|  |  |  |             raise ValueError(errmsg("Expecting property name", s, end - 1)) | 
					
						
							| 
									
										
										
										
											2009-04-21 03:09:17 +00:00
										 |  |  |     if object_pairs_hook is not None: | 
					
						
							|  |  |  |         result = object_pairs_hook(pairs) | 
					
						
							|  |  |  |         return result, end | 
					
						
							|  |  |  |     pairs = dict(pairs) | 
					
						
							| 
									
										
										
										
											2008-05-08 14:29:10 +00:00
										 |  |  |     if object_hook is not None: | 
					
						
							|  |  |  |         pairs = object_hook(pairs) | 
					
						
							|  |  |  |     return pairs, end | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-09-04 20:16:53 +00:00
										 |  |  | def JSONArray(s_and_end, scan_once, _w=WHITESPACE.match, _ws=WHITESPACE_STR): | 
					
						
							| 
									
										
										
										
											2009-05-02 12:36:44 +00:00
										 |  |  |     s, end = s_and_end | 
					
						
							| 
									
										
										
										
											2008-05-08 14:29:10 +00:00
										 |  |  |     values = [] | 
					
						
							|  |  |  |     nextchar = s[end:end + 1] | 
					
						
							| 
									
										
										
										
											2009-05-02 12:36:44 +00:00
										 |  |  |     if nextchar in _ws: | 
					
						
							|  |  |  |         end = _w(s, end + 1).end() | 
					
						
							|  |  |  |         nextchar = s[end:end + 1] | 
					
						
							|  |  |  |     # Look-ahead for trivial empty array | 
					
						
							| 
									
										
										
										
											2008-05-08 14:29:10 +00:00
										 |  |  |     if nextchar == ']': | 
					
						
							|  |  |  |         return values, end + 1 | 
					
						
							| 
									
										
										
										
											2009-05-02 12:36:44 +00:00
										 |  |  |     _append = values.append | 
					
						
							| 
									
										
										
										
											2008-05-08 14:29:10 +00:00
										 |  |  |     while True: | 
					
						
							|  |  |  |         try: | 
					
						
							| 
									
										
										
										
											2009-05-02 12:36:44 +00:00
										 |  |  |             value, end = scan_once(s, end) | 
					
						
							| 
									
										
										
										
											2008-05-08 14:29:10 +00:00
										 |  |  |         except StopIteration: | 
					
						
							|  |  |  |             raise ValueError(errmsg("Expecting object", s, end)) | 
					
						
							| 
									
										
										
										
											2009-05-02 12:36:44 +00:00
										 |  |  |         _append(value) | 
					
						
							| 
									
										
										
										
											2008-05-08 14:29:10 +00:00
										 |  |  |         nextchar = s[end:end + 1] | 
					
						
							| 
									
										
										
										
											2009-05-02 12:36:44 +00:00
										 |  |  |         if nextchar in _ws: | 
					
						
							|  |  |  |             end = _w(s, end + 1).end() | 
					
						
							|  |  |  |             nextchar = s[end:end + 1] | 
					
						
							| 
									
										
										
										
											2008-05-08 14:29:10 +00:00
										 |  |  |         end += 1 | 
					
						
							|  |  |  |         if nextchar == ']': | 
					
						
							|  |  |  |             break | 
					
						
							| 
									
										
										
										
											2009-05-02 12:36:44 +00:00
										 |  |  |         elif nextchar != ',': | 
					
						
							| 
									
										
										
										
											2008-05-08 14:29:10 +00:00
										 |  |  |             raise ValueError(errmsg("Expecting , delimiter", s, end)) | 
					
						
							| 
									
										
										
										
											2009-05-02 12:36:44 +00:00
										 |  |  |         try: | 
					
						
							|  |  |  |             if s[end] in _ws: | 
					
						
							|  |  |  |                 end += 1 | 
					
						
							|  |  |  |                 if s[end] in _ws: | 
					
						
							|  |  |  |                     end = _w(s, end + 1).end() | 
					
						
							|  |  |  |         except IndexError: | 
					
						
							|  |  |  |             pass | 
					
						
							| 
									
										
										
										
											2008-05-08 14:29:10 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-05-02 12:36:44 +00:00
										 |  |  |     return values, end | 
					
						
							| 
									
										
										
										
											2008-05-08 14:29:10 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | class JSONDecoder(object): | 
					
						
							|  |  |  |     """Simple JSON <http://json.org> decoder
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     Performs the following translations in decoding by default: | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     +---------------+-------------------+ | 
					
						
							|  |  |  |     | JSON          | Python            | | 
					
						
							|  |  |  |     +===============+===================+ | 
					
						
							|  |  |  |     | object        | dict              | | 
					
						
							|  |  |  |     +---------------+-------------------+ | 
					
						
							|  |  |  |     | array         | list              | | 
					
						
							|  |  |  |     +---------------+-------------------+ | 
					
						
							| 
									
										
										
										
											2010-08-02 20:16:18 +00:00
										 |  |  |     | string        | str               | | 
					
						
							| 
									
										
										
										
											2008-05-08 14:29:10 +00:00
										 |  |  |     +---------------+-------------------+ | 
					
						
							| 
									
										
										
										
											2010-08-02 20:16:18 +00:00
										 |  |  |     | number (int)  | int               | | 
					
						
							| 
									
										
										
										
											2008-05-08 14:29:10 +00:00
										 |  |  |     +---------------+-------------------+ | 
					
						
							|  |  |  |     | number (real) | float             | | 
					
						
							|  |  |  |     +---------------+-------------------+ | 
					
						
							|  |  |  |     | true          | True              | | 
					
						
							|  |  |  |     +---------------+-------------------+ | 
					
						
							|  |  |  |     | false         | False             | | 
					
						
							|  |  |  |     +---------------+-------------------+ | 
					
						
							|  |  |  |     | null          | None              | | 
					
						
							|  |  |  |     +---------------+-------------------+ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     It also understands ``NaN``, ``Infinity``, and ``-Infinity`` as | 
					
						
							|  |  |  |     their corresponding ``float`` values, which is outside the JSON spec. | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-05-02 12:36:44 +00:00
										 |  |  |     """
 | 
					
						
							| 
									
										
										
										
											2008-05-08 14:29:10 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-05-02 12:36:44 +00:00
										 |  |  |     def __init__(self, object_hook=None, parse_float=None, | 
					
						
							| 
									
										
										
										
											2009-04-21 03:09:17 +00:00
										 |  |  |             parse_int=None, parse_constant=None, strict=True, | 
					
						
							|  |  |  |             object_pairs_hook=None): | 
					
						
							| 
									
										
										
										
											2009-05-02 12:36:44 +00:00
										 |  |  |         """``object_hook``, if specified, will be called with the result
 | 
					
						
							|  |  |  |         of every JSON object decoded and its return value will be used in | 
					
						
							| 
									
										
										
										
											2008-05-08 14:29:10 +00:00
										 |  |  |         place of the given ``dict``.  This can be used to provide custom | 
					
						
							|  |  |  |         deserializations (e.g. to support JSON-RPC class hinting). | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-10-15 17:03:02 +00:00
										 |  |  |         ``object_pairs_hook``, if specified will be called with the result of | 
					
						
							|  |  |  |         every JSON object decoded with an ordered list of pairs.  The return | 
					
						
							|  |  |  |         value of ``object_pairs_hook`` will be used instead of the ``dict``. | 
					
						
							|  |  |  |         This feature can be used to implement custom decoders that rely on the | 
					
						
							|  |  |  |         order that the key and value pairs are decoded (for example, | 
					
						
							|  |  |  |         collections.OrderedDict will remember the order of insertion). If | 
					
						
							|  |  |  |         ``object_hook`` is also defined, the ``object_pairs_hook`` takes | 
					
						
							|  |  |  |         priority. | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-05-08 14:29:10 +00:00
										 |  |  |         ``parse_float``, if specified, will be called with the string | 
					
						
							|  |  |  |         of every JSON float to be decoded. By default this is equivalent to | 
					
						
							|  |  |  |         float(num_str). This can be used to use another datatype or parser | 
					
						
							|  |  |  |         for JSON floats (e.g. decimal.Decimal). | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         ``parse_int``, if specified, will be called with the string | 
					
						
							|  |  |  |         of every JSON int to be decoded. By default this is equivalent to | 
					
						
							|  |  |  |         int(num_str). This can be used to use another datatype or parser | 
					
						
							|  |  |  |         for JSON integers (e.g. float). | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         ``parse_constant``, if specified, will be called with one of the | 
					
						
							| 
									
										
										
										
											2009-05-02 12:36:44 +00:00
										 |  |  |         following strings: -Infinity, Infinity, NaN. | 
					
						
							| 
									
										
										
										
											2008-05-08 14:29:10 +00:00
										 |  |  |         This can be used to raise an exception if invalid JSON numbers | 
					
						
							|  |  |  |         are encountered. | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-10-15 17:03:02 +00:00
										 |  |  |         If ``strict`` is false (true is the default), then control | 
					
						
							|  |  |  |         characters will be allowed inside strings.  Control characters in | 
					
						
							|  |  |  |         this context are those with character codes in the 0-31 range, | 
					
						
							|  |  |  |         including ``'\\t'`` (tab), ``'\\n'``, ``'\\r'`` and ``'\\0'``. | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-05-08 14:29:10 +00:00
										 |  |  |         """
 | 
					
						
							|  |  |  |         self.object_hook = object_hook | 
					
						
							| 
									
										
										
										
											2009-05-02 12:36:44 +00:00
										 |  |  |         self.parse_float = parse_float or float | 
					
						
							|  |  |  |         self.parse_int = parse_int or int | 
					
						
							|  |  |  |         self.parse_constant = parse_constant or _CONSTANTS.__getitem__ | 
					
						
							| 
									
										
										
										
											2008-05-08 14:29:10 +00:00
										 |  |  |         self.strict = strict | 
					
						
							| 
									
										
										
										
											2009-05-02 12:36:44 +00:00
										 |  |  |         self.object_pairs_hook = object_pairs_hook | 
					
						
							|  |  |  |         self.parse_object = JSONObject | 
					
						
							|  |  |  |         self.parse_array = JSONArray | 
					
						
							|  |  |  |         self.parse_string = scanstring | 
					
						
							| 
									
										
										
										
											2010-09-04 20:16:53 +00:00
										 |  |  |         self.memo = {} | 
					
						
							| 
									
										
										
										
											2011-05-14 06:47:51 +03:00
										 |  |  |         self.scan_once = scanner.make_scanner(self) | 
					
						
							| 
									
										
										
										
											2009-05-02 12:36:44 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-05-08 14:29:10 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |     def decode(self, s, _w=WHITESPACE.match): | 
					
						
							| 
									
										
										
										
											2010-08-02 20:16:18 +00:00
										 |  |  |         """Return the Python representation of ``s`` (a ``str`` instance
 | 
					
						
							|  |  |  |         containing a JSON document). | 
					
						
							| 
									
										
										
										
											2008-05-08 14:29:10 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |         """
 | 
					
						
							|  |  |  |         obj, end = self.raw_decode(s, idx=_w(s, 0).end()) | 
					
						
							|  |  |  |         end = _w(s, end).end() | 
					
						
							|  |  |  |         if end != len(s): | 
					
						
							|  |  |  |             raise ValueError(errmsg("Extra data", s, end, len(s))) | 
					
						
							|  |  |  |         return obj | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-05-02 12:36:44 +00:00
										 |  |  |     def raw_decode(self, s, idx=0): | 
					
						
							| 
									
										
										
										
											2010-08-02 20:16:18 +00:00
										 |  |  |         """Decode a JSON document from ``s`` (a ``str`` beginning with
 | 
					
						
							|  |  |  |         a JSON document) and return a 2-tuple of the Python | 
					
						
							| 
									
										
										
										
											2008-05-08 14:29:10 +00:00
										 |  |  |         representation and the index in ``s`` where the document ended. | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         This can be used to decode a JSON document from a string that may | 
					
						
							|  |  |  |         have extraneous data at the end. | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         """
 | 
					
						
							|  |  |  |         try: | 
					
						
							| 
									
										
										
										
											2009-05-02 12:36:44 +00:00
										 |  |  |             obj, end = self.scan_once(s, idx) | 
					
						
							| 
									
										
										
										
											2008-05-08 14:29:10 +00:00
										 |  |  |         except StopIteration: | 
					
						
							|  |  |  |             raise ValueError("No JSON object could be decoded") | 
					
						
							|  |  |  |         return obj, end |