| 
									
										
										
										
											2019-04-17 23:02:26 +02:00
										 |  |  | /* JSON accelerator C extensor: _json module.
 | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * It is built as a built-in module (Py_BUILD_CORE_BUILTIN define) on Windows | 
					
						
							|  |  |  |  * and as an extension module (Py_BUILD_CORE_MODULE define) on other | 
					
						
							|  |  |  |  * platforms. */ | 
					
						
							| 
									
										
										
										
											2017-09-07 23:51:28 -06:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-10-22 16:36:28 +03:00
										 |  |  | #ifndef Py_BUILD_CORE_BUILTIN
 | 
					
						
							|  |  |  | #  define Py_BUILD_CORE_MODULE 1
 | 
					
						
							| 
									
										
										
										
											2017-09-07 23:51:28 -06:00
										 |  |  | #endif
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-05-08 14:29:10 +00:00
										 |  |  | #include "Python.h"
 | 
					
						
							| 
									
										
										
										
											2022-11-08 10:03:03 -07:00
										 |  |  | #include "pycore_ceval.h"           // _Py_EnterRecursiveCall()
 | 
					
						
							|  |  |  | #include "pycore_runtime.h"         // _PyRuntime
 | 
					
						
							|  |  |  | #include "structmember.h"           // PyMemberDef
 | 
					
						
							|  |  |  | #include "pycore_global_objects.h"  // _Py_ID()
 | 
					
						
							|  |  |  | #include <stdbool.h>                // bool
 | 
					
						
							| 
									
										
										
										
											2012-03-22 14:42:18 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-05-02 12:36:44 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | typedef struct _PyScannerObject { | 
					
						
							|  |  |  |     PyObject_HEAD | 
					
						
							| 
									
										
										
										
											2017-07-13 08:52:08 +03:00
										 |  |  |     signed char strict; | 
					
						
							| 
									
										
										
										
											2009-05-02 12:36:44 +00:00
										 |  |  |     PyObject *object_hook; | 
					
						
							|  |  |  |     PyObject *object_pairs_hook; | 
					
						
							|  |  |  |     PyObject *parse_float; | 
					
						
							|  |  |  |     PyObject *parse_int; | 
					
						
							|  |  |  |     PyObject *parse_constant; | 
					
						
							| 
									
										
										
										
											2010-09-04 20:16:53 +00:00
										 |  |  |     PyObject *memo; | 
					
						
							| 
									
										
										
										
											2009-05-02 12:36:44 +00:00
										 |  |  | } PyScannerObject; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static PyMemberDef scanner_members[] = { | 
					
						
							| 
									
										
										
										
											2017-05-28 15:31:49 +03:00
										 |  |  |     {"strict", T_BOOL, offsetof(PyScannerObject, strict), READONLY, "strict"}, | 
					
						
							| 
									
										
										
										
											2009-05-02 12:36:44 +00:00
										 |  |  |     {"object_hook", T_OBJECT, offsetof(PyScannerObject, object_hook), READONLY, "object_hook"}, | 
					
						
							|  |  |  |     {"object_pairs_hook", T_OBJECT, offsetof(PyScannerObject, object_pairs_hook), READONLY}, | 
					
						
							|  |  |  |     {"parse_float", T_OBJECT, offsetof(PyScannerObject, parse_float), READONLY, "parse_float"}, | 
					
						
							|  |  |  |     {"parse_int", T_OBJECT, offsetof(PyScannerObject, parse_int), READONLY, "parse_int"}, | 
					
						
							|  |  |  |     {"parse_constant", T_OBJECT, offsetof(PyScannerObject, parse_constant), READONLY, "parse_constant"}, | 
					
						
							|  |  |  |     {NULL} | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | typedef struct _PyEncoderObject { | 
					
						
							|  |  |  |     PyObject_HEAD | 
					
						
							|  |  |  |     PyObject *markers; | 
					
						
							|  |  |  |     PyObject *defaultfn; | 
					
						
							|  |  |  |     PyObject *encoder; | 
					
						
							|  |  |  |     PyObject *indent; | 
					
						
							|  |  |  |     PyObject *key_separator; | 
					
						
							|  |  |  |     PyObject *item_separator; | 
					
						
							| 
									
										
										
										
											2017-05-28 15:31:49 +03:00
										 |  |  |     char sort_keys; | 
					
						
							|  |  |  |     char skipkeys; | 
					
						
							| 
									
										
										
										
											2009-05-02 12:36:44 +00:00
										 |  |  |     int allow_nan; | 
					
						
							| 
									
										
										
										
											2017-05-28 15:31:49 +03:00
										 |  |  |     PyCFunction fast_encode; | 
					
						
							| 
									
										
										
										
											2009-05-02 12:36:44 +00:00
										 |  |  | } PyEncoderObject; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static PyMemberDef encoder_members[] = { | 
					
						
							|  |  |  |     {"markers", T_OBJECT, offsetof(PyEncoderObject, markers), READONLY, "markers"}, | 
					
						
							|  |  |  |     {"default", T_OBJECT, offsetof(PyEncoderObject, defaultfn), READONLY, "default"}, | 
					
						
							|  |  |  |     {"encoder", T_OBJECT, offsetof(PyEncoderObject, encoder), READONLY, "encoder"}, | 
					
						
							|  |  |  |     {"indent", T_OBJECT, offsetof(PyEncoderObject, indent), READONLY, "indent"}, | 
					
						
							|  |  |  |     {"key_separator", T_OBJECT, offsetof(PyEncoderObject, key_separator), READONLY, "key_separator"}, | 
					
						
							|  |  |  |     {"item_separator", T_OBJECT, offsetof(PyEncoderObject, item_separator), READONLY, "item_separator"}, | 
					
						
							| 
									
										
										
										
											2017-05-28 15:31:49 +03:00
										 |  |  |     {"sort_keys", T_BOOL, offsetof(PyEncoderObject, sort_keys), READONLY, "sort_keys"}, | 
					
						
							|  |  |  |     {"skipkeys", T_BOOL, offsetof(PyEncoderObject, skipkeys), READONLY, "skipkeys"}, | 
					
						
							| 
									
										
										
										
											2009-05-02 12:36:44 +00:00
										 |  |  |     {NULL} | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-08-19 18:03:14 +02:00
										 |  |  | /* Forward decls */ | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-05-02 12:36:44 +00:00
										 |  |  | static PyObject * | 
					
						
							|  |  |  | ascii_escape_unicode(PyObject *pystr); | 
					
						
							|  |  |  | static PyObject * | 
					
						
							| 
									
										
										
										
											2019-11-05 11:44:28 +01:00
										 |  |  | py_encode_basestring_ascii(PyObject* Py_UNUSED(self), PyObject *pystr); | 
					
						
							| 
									
										
										
										
											2009-05-02 12:36:44 +00:00
										 |  |  | static PyObject * | 
					
						
							|  |  |  | scan_once_unicode(PyScannerObject *s, PyObject *pystr, Py_ssize_t idx, Py_ssize_t *next_idx_ptr); | 
					
						
							|  |  |  | static PyObject * | 
					
						
							|  |  |  | _build_rval_index_tuple(PyObject *rval, Py_ssize_t idx); | 
					
						
							|  |  |  | static PyObject * | 
					
						
							|  |  |  | scanner_new(PyTypeObject *type, PyObject *args, PyObject *kwds); | 
					
						
							|  |  |  | static void | 
					
						
							|  |  |  | scanner_dealloc(PyObject *self); | 
					
						
							|  |  |  | static int | 
					
						
							| 
									
										
										
										
											2020-03-27 19:59:59 +09:00
										 |  |  | scanner_clear(PyScannerObject *self); | 
					
						
							| 
									
										
										
										
											2009-05-02 12:36:44 +00:00
										 |  |  | static PyObject * | 
					
						
							|  |  |  | encoder_new(PyTypeObject *type, PyObject *args, PyObject *kwds); | 
					
						
							|  |  |  | static void | 
					
						
							|  |  |  | encoder_dealloc(PyObject *self); | 
					
						
							|  |  |  | static int | 
					
						
							| 
									
										
										
										
											2020-03-27 19:59:59 +09:00
										 |  |  | encoder_clear(PyEncoderObject *self); | 
					
						
							| 
									
										
										
										
											2009-05-02 12:36:44 +00:00
										 |  |  | static int | 
					
						
							| 
									
										
										
										
											2022-07-27 11:43:34 +03:00
										 |  |  | encoder_listencode_list(PyEncoderObject *s, _PyUnicodeWriter *writer, PyObject *seq, Py_ssize_t indent_level); | 
					
						
							| 
									
										
										
										
											2009-05-02 12:36:44 +00:00
										 |  |  | static int | 
					
						
							| 
									
										
										
										
											2022-07-27 11:43:34 +03:00
										 |  |  | encoder_listencode_obj(PyEncoderObject *s, _PyUnicodeWriter *writer, PyObject *obj, Py_ssize_t indent_level); | 
					
						
							| 
									
										
										
										
											2009-05-02 12:36:44 +00:00
										 |  |  | static int | 
					
						
							| 
									
										
										
										
											2022-07-27 11:43:34 +03:00
										 |  |  | encoder_listencode_dict(PyEncoderObject *s, _PyUnicodeWriter *writer, PyObject *dct, Py_ssize_t indent_level); | 
					
						
							| 
									
										
										
										
											2009-05-02 12:36:44 +00:00
										 |  |  | static PyObject * | 
					
						
							| 
									
										
										
										
											2009-05-02 15:55:19 +00:00
										 |  |  | _encoded_const(PyObject *obj); | 
					
						
							| 
									
										
										
										
											2009-05-02 12:36:44 +00:00
										 |  |  | static void | 
					
						
							| 
									
										
										
										
											2015-12-25 20:01:53 +02:00
										 |  |  | raise_errmsg(const char *msg, PyObject *s, Py_ssize_t end); | 
					
						
							| 
									
										
										
										
											2009-05-02 12:36:44 +00:00
										 |  |  | static PyObject * | 
					
						
							|  |  |  | encoder_encode_string(PyEncoderObject *s, PyObject *obj); | 
					
						
							|  |  |  | static PyObject * | 
					
						
							|  |  |  | encoder_encode_float(PyEncoderObject *s, PyObject *obj); | 
					
						
							| 
									
										
										
										
											2008-05-08 14:29:10 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | #define S_CHAR(c) (c >= ' ' && c <= '~' && c != '\\' && c != '"')
 | 
					
						
							| 
									
										
										
										
											2009-05-02 12:36:44 +00:00
										 |  |  | #define IS_WHITESPACE(c) (((c) == ' ') || ((c) == '\t') || ((c) == '\n') || ((c) == '\r'))
 | 
					
						
							| 
									
										
										
										
											2008-05-08 14:29:10 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | static Py_ssize_t | 
					
						
							| 
									
										
										
										
											2011-09-28 07:41:54 +02:00
										 |  |  | ascii_escape_unichar(Py_UCS4 c, unsigned char *output, Py_ssize_t chars) | 
					
						
							| 
									
										
										
										
											2008-05-08 14:29:10 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2009-05-02 12:36:44 +00:00
										 |  |  |     /* Escape unicode code point c to ASCII escape sequences
 | 
					
						
							|  |  |  |     in char *output. output must have at least 12 bytes unused to | 
					
						
							|  |  |  |     accommodate an escaped surrogate pair "\uXXXX\uXXXX" */ | 
					
						
							| 
									
										
										
										
											2008-05-08 14:29:10 +00:00
										 |  |  |     output[chars++] = '\\'; | 
					
						
							|  |  |  |     switch (c) { | 
					
						
							| 
									
										
										
										
											2009-05-02 12:36:44 +00:00
										 |  |  |         case '\\': output[chars++] = c; break; | 
					
						
							|  |  |  |         case '"': output[chars++] = c; break; | 
					
						
							| 
									
										
										
										
											2008-05-08 14:29:10 +00:00
										 |  |  |         case '\b': output[chars++] = 'b'; break; | 
					
						
							|  |  |  |         case '\f': output[chars++] = 'f'; break; | 
					
						
							|  |  |  |         case '\n': output[chars++] = 'n'; break; | 
					
						
							|  |  |  |         case '\r': output[chars++] = 'r'; break; | 
					
						
							|  |  |  |         case '\t': output[chars++] = 't'; break; | 
					
						
							|  |  |  |         default: | 
					
						
							|  |  |  |             if (c >= 0x10000) { | 
					
						
							|  |  |  |                 /* UTF-16 surrogate pair */ | 
					
						
							| 
									
										
										
										
											2012-10-30 01:42:39 +01:00
										 |  |  |                 Py_UCS4 v = Py_UNICODE_HIGH_SURROGATE(c); | 
					
						
							| 
									
										
										
										
											2008-05-08 14:29:10 +00:00
										 |  |  |                 output[chars++] = 'u'; | 
					
						
							| 
									
										
										
										
											2012-10-30 01:42:39 +01:00
										 |  |  |                 output[chars++] = Py_hexdigits[(v >> 12) & 0xf]; | 
					
						
							|  |  |  |                 output[chars++] = Py_hexdigits[(v >>  8) & 0xf]; | 
					
						
							|  |  |  |                 output[chars++] = Py_hexdigits[(v >>  4) & 0xf]; | 
					
						
							|  |  |  |                 output[chars++] = Py_hexdigits[(v      ) & 0xf]; | 
					
						
							|  |  |  |                 c = Py_UNICODE_LOW_SURROGATE(c); | 
					
						
							| 
									
										
										
										
											2008-05-08 14:29:10 +00:00
										 |  |  |                 output[chars++] = '\\'; | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |             output[chars++] = 'u'; | 
					
						
							| 
									
										
										
										
											2011-10-14 02:13:11 +02:00
										 |  |  |             output[chars++] = Py_hexdigits[(c >> 12) & 0xf]; | 
					
						
							|  |  |  |             output[chars++] = Py_hexdigits[(c >>  8) & 0xf]; | 
					
						
							|  |  |  |             output[chars++] = Py_hexdigits[(c >>  4) & 0xf]; | 
					
						
							|  |  |  |             output[chars++] = Py_hexdigits[(c      ) & 0xf]; | 
					
						
							| 
									
										
										
										
											2008-05-08 14:29:10 +00:00
										 |  |  |     } | 
					
						
							|  |  |  |     return chars; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static PyObject * | 
					
						
							|  |  |  | ascii_escape_unicode(PyObject *pystr) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2009-05-02 12:36:44 +00:00
										 |  |  |     /* Take a PyUnicode pystr and return a new ASCII-only escaped PyUnicode */ | 
					
						
							| 
									
										
										
										
											2008-05-08 14:29:10 +00:00
										 |  |  |     Py_ssize_t i; | 
					
						
							|  |  |  |     Py_ssize_t input_chars; | 
					
						
							|  |  |  |     Py_ssize_t output_size; | 
					
						
							|  |  |  |     Py_ssize_t chars; | 
					
						
							|  |  |  |     PyObject *rval; | 
					
						
							| 
									
										
										
										
											2020-04-11 10:48:40 +03:00
										 |  |  |     const void *input; | 
					
						
							|  |  |  |     Py_UCS1 *output; | 
					
						
							| 
									
										
										
										
											2011-09-28 07:41:54 +02:00
										 |  |  |     int kind; | 
					
						
							| 
									
										
										
										
											2008-05-08 14:29:10 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-09-28 07:41:54 +02:00
										 |  |  |     if (PyUnicode_READY(pystr) == -1) | 
					
						
							|  |  |  |         return NULL; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     input_chars = PyUnicode_GET_LENGTH(pystr); | 
					
						
							|  |  |  |     input = PyUnicode_DATA(pystr); | 
					
						
							|  |  |  |     kind = PyUnicode_KIND(pystr); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     /* Compute the output size */ | 
					
						
							|  |  |  |     for (i = 0, output_size = 2; i < input_chars; i++) { | 
					
						
							|  |  |  |         Py_UCS4 c = PyUnicode_READ(kind, input, i); | 
					
						
							| 
									
										
										
										
											2015-02-01 17:53:53 -05:00
										 |  |  |         Py_ssize_t d; | 
					
						
							|  |  |  |         if (S_CHAR(c)) { | 
					
						
							|  |  |  |             d = 1; | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											2011-09-28 07:41:54 +02:00
										 |  |  |         else { | 
					
						
							|  |  |  |             switch(c) { | 
					
						
							| 
									
										
										
										
											2011-10-11 21:56:19 +02:00
										 |  |  |             case '\\': case '"': case '\b': case '\f': | 
					
						
							| 
									
										
										
										
											2011-09-28 07:41:54 +02:00
										 |  |  |             case '\n': case '\r': case '\t': | 
					
						
							| 
									
										
										
										
											2015-02-01 17:53:53 -05:00
										 |  |  |                 d = 2; break; | 
					
						
							| 
									
										
										
										
											2011-09-28 07:41:54 +02:00
										 |  |  |             default: | 
					
						
							| 
									
										
										
										
											2015-02-01 17:53:53 -05:00
										 |  |  |                 d = c >= 0x10000 ? 12 : 6; | 
					
						
							| 
									
										
										
										
											2011-09-28 07:41:54 +02:00
										 |  |  |             } | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											2015-02-01 17:53:53 -05:00
										 |  |  |         if (output_size > PY_SSIZE_T_MAX - d) { | 
					
						
							|  |  |  |             PyErr_SetString(PyExc_OverflowError, "string is too long to escape"); | 
					
						
							|  |  |  |             return NULL; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         output_size += d; | 
					
						
							| 
									
										
										
										
											2011-09-28 07:41:54 +02:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2009-05-02 12:36:44 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-09-28 07:41:54 +02:00
										 |  |  |     rval = PyUnicode_New(output_size, 127); | 
					
						
							| 
									
										
										
										
											2008-05-08 14:29:10 +00:00
										 |  |  |     if (rval == NULL) { | 
					
						
							|  |  |  |         return NULL; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2011-09-28 07:41:54 +02:00
										 |  |  |     output = PyUnicode_1BYTE_DATA(rval); | 
					
						
							| 
									
										
										
										
											2008-05-08 14:29:10 +00:00
										 |  |  |     chars = 0; | 
					
						
							|  |  |  |     output[chars++] = '"'; | 
					
						
							|  |  |  |     for (i = 0; i < input_chars; i++) { | 
					
						
							| 
									
										
										
										
											2011-09-28 07:41:54 +02:00
										 |  |  |         Py_UCS4 c = PyUnicode_READ(kind, input, i); | 
					
						
							| 
									
										
										
										
											2008-05-08 14:29:10 +00:00
										 |  |  |         if (S_CHAR(c)) { | 
					
						
							| 
									
										
										
										
											2009-05-02 12:36:44 +00:00
										 |  |  |             output[chars++] = c; | 
					
						
							| 
									
										
										
										
											2008-05-08 14:29:10 +00:00
										 |  |  |         } | 
					
						
							| 
									
										
										
										
											2009-05-02 12:36:44 +00:00
										 |  |  |         else { | 
					
						
							|  |  |  |             chars = ascii_escape_unichar(c, output, chars); | 
					
						
							| 
									
										
										
										
											2008-05-08 14:29:10 +00:00
										 |  |  |         } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     output[chars++] = '"'; | 
					
						
							| 
									
										
										
										
											2013-01-03 09:21:55 +01:00
										 |  |  | #ifdef Py_DEBUG
 | 
					
						
							| 
									
										
										
										
											2012-04-27 13:55:39 +02:00
										 |  |  |     assert(_PyUnicode_CheckConsistency(rval, 1)); | 
					
						
							| 
									
										
										
										
											2013-01-03 09:21:55 +01:00
										 |  |  | #endif
 | 
					
						
							| 
									
										
										
										
											2008-05-08 14:29:10 +00:00
										 |  |  |     return rval; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-01-11 16:41:01 +01:00
										 |  |  | static PyObject * | 
					
						
							|  |  |  | escape_unicode(PyObject *pystr) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     /* Take a PyUnicode pystr and return a new escaped PyUnicode */ | 
					
						
							|  |  |  |     Py_ssize_t i; | 
					
						
							|  |  |  |     Py_ssize_t input_chars; | 
					
						
							|  |  |  |     Py_ssize_t output_size; | 
					
						
							|  |  |  |     Py_ssize_t chars; | 
					
						
							|  |  |  |     PyObject *rval; | 
					
						
							| 
									
										
										
										
											2020-04-11 10:48:40 +03:00
										 |  |  |     const void *input; | 
					
						
							| 
									
										
										
										
											2015-01-11 16:41:01 +01:00
										 |  |  |     int kind; | 
					
						
							|  |  |  |     Py_UCS4 maxchar; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (PyUnicode_READY(pystr) == -1) | 
					
						
							|  |  |  |         return NULL; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     maxchar = PyUnicode_MAX_CHAR_VALUE(pystr); | 
					
						
							|  |  |  |     input_chars = PyUnicode_GET_LENGTH(pystr); | 
					
						
							|  |  |  |     input = PyUnicode_DATA(pystr); | 
					
						
							|  |  |  |     kind = PyUnicode_KIND(pystr); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     /* Compute the output size */ | 
					
						
							|  |  |  |     for (i = 0, output_size = 2; i < input_chars; i++) { | 
					
						
							|  |  |  |         Py_UCS4 c = PyUnicode_READ(kind, input, i); | 
					
						
							| 
									
										
										
										
											2015-06-27 15:01:51 -05:00
										 |  |  |         Py_ssize_t d; | 
					
						
							| 
									
										
										
										
											2015-01-11 16:41:01 +01:00
										 |  |  |         switch (c) { | 
					
						
							|  |  |  |         case '\\': case '"': case '\b': case '\f': | 
					
						
							|  |  |  |         case '\n': case '\r': case '\t': | 
					
						
							| 
									
										
										
										
											2015-06-27 15:01:51 -05:00
										 |  |  |             d = 2; | 
					
						
							| 
									
										
										
										
											2015-01-11 16:41:01 +01:00
										 |  |  |             break; | 
					
						
							|  |  |  |         default: | 
					
						
							|  |  |  |             if (c <= 0x1f) | 
					
						
							| 
									
										
										
										
											2015-06-27 15:01:51 -05:00
										 |  |  |                 d = 6; | 
					
						
							| 
									
										
										
										
											2015-01-11 16:41:01 +01:00
										 |  |  |             else | 
					
						
							| 
									
										
										
										
											2015-06-27 15:01:51 -05:00
										 |  |  |                 d = 1; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         if (output_size > PY_SSIZE_T_MAX - d) { | 
					
						
							|  |  |  |             PyErr_SetString(PyExc_OverflowError, "string is too long to escape"); | 
					
						
							|  |  |  |             return NULL; | 
					
						
							| 
									
										
										
										
											2015-01-11 16:41:01 +01:00
										 |  |  |         } | 
					
						
							| 
									
										
										
										
											2015-06-27 15:01:51 -05:00
										 |  |  |         output_size += d; | 
					
						
							| 
									
										
										
										
											2015-01-11 16:41:01 +01:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     rval = PyUnicode_New(output_size, maxchar); | 
					
						
							|  |  |  |     if (rval == NULL) | 
					
						
							|  |  |  |         return NULL; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     kind = PyUnicode_KIND(rval); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #define ENCODE_OUTPUT do { \
 | 
					
						
							|  |  |  |         chars = 0; \ | 
					
						
							|  |  |  |         output[chars++] = '"'; \ | 
					
						
							|  |  |  |         for (i = 0; i < input_chars; i++) { \ | 
					
						
							|  |  |  |             Py_UCS4 c = PyUnicode_READ(kind, input, i); \ | 
					
						
							|  |  |  |             switch (c) { \ | 
					
						
							|  |  |  |             case '\\': output[chars++] = '\\'; output[chars++] = c; break; \ | 
					
						
							|  |  |  |             case '"':  output[chars++] = '\\'; output[chars++] = c; break; \ | 
					
						
							|  |  |  |             case '\b': output[chars++] = '\\'; output[chars++] = 'b'; break; \ | 
					
						
							|  |  |  |             case '\f': output[chars++] = '\\'; output[chars++] = 'f'; break; \ | 
					
						
							|  |  |  |             case '\n': output[chars++] = '\\'; output[chars++] = 'n'; break; \ | 
					
						
							|  |  |  |             case '\r': output[chars++] = '\\'; output[chars++] = 'r'; break; \ | 
					
						
							|  |  |  |             case '\t': output[chars++] = '\\'; output[chars++] = 't'; break; \ | 
					
						
							|  |  |  |             default: \ | 
					
						
							|  |  |  |                 if (c <= 0x1f) { \ | 
					
						
							|  |  |  |                     output[chars++] = '\\'; \ | 
					
						
							|  |  |  |                     output[chars++] = 'u'; \ | 
					
						
							|  |  |  |                     output[chars++] = '0'; \ | 
					
						
							|  |  |  |                     output[chars++] = '0'; \ | 
					
						
							|  |  |  |                     output[chars++] = Py_hexdigits[(c >> 4) & 0xf]; \ | 
					
						
							|  |  |  |                     output[chars++] = Py_hexdigits[(c     ) & 0xf]; \ | 
					
						
							|  |  |  |                 } else { \ | 
					
						
							|  |  |  |                     output[chars++] = c; \ | 
					
						
							|  |  |  |                 } \ | 
					
						
							|  |  |  |             } \ | 
					
						
							|  |  |  |         } \ | 
					
						
							|  |  |  |         output[chars++] = '"'; \ | 
					
						
							|  |  |  |     } while (0) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (kind == PyUnicode_1BYTE_KIND) { | 
					
						
							|  |  |  |         Py_UCS1 *output = PyUnicode_1BYTE_DATA(rval); | 
					
						
							|  |  |  |         ENCODE_OUTPUT; | 
					
						
							|  |  |  |     } else if (kind == PyUnicode_2BYTE_KIND) { | 
					
						
							|  |  |  |         Py_UCS2 *output = PyUnicode_2BYTE_DATA(rval); | 
					
						
							|  |  |  |         ENCODE_OUTPUT; | 
					
						
							|  |  |  |     } else { | 
					
						
							|  |  |  |         Py_UCS4 *output = PyUnicode_4BYTE_DATA(rval); | 
					
						
							|  |  |  |         assert(kind == PyUnicode_4BYTE_KIND); | 
					
						
							|  |  |  |         ENCODE_OUTPUT; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | #undef ENCODE_OUTPUT
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #ifdef Py_DEBUG
 | 
					
						
							|  |  |  |     assert(_PyUnicode_CheckConsistency(rval, 1)); | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  |     return rval; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-05-02 12:36:44 +00:00
										 |  |  | static void | 
					
						
							| 
									
										
										
										
											2015-12-25 20:01:53 +02:00
										 |  |  | raise_errmsg(const char *msg, PyObject *s, Py_ssize_t end) | 
					
						
							| 
									
										
										
										
											2008-05-08 14:29:10 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2015-01-26 13:16:30 +02:00
										 |  |  |     /* Use JSONDecodeError exception to raise a nice looking ValueError subclass */ | 
					
						
							| 
									
										
										
										
											2022-11-02 21:33:38 +05:30
										 |  |  |     _Py_DECLARE_STR(json_decoder, "json.decoder"); | 
					
						
							|  |  |  |     PyObject *JSONDecodeError = | 
					
						
							|  |  |  |          _PyImport_GetModuleAttr(&_Py_STR(json_decoder), &_Py_ID(JSONDecodeError)); | 
					
						
							| 
									
										
										
										
											2015-01-26 13:16:30 +02:00
										 |  |  |     if (JSONDecodeError == NULL) { | 
					
						
							| 
									
										
										
										
											2021-02-02 00:26:56 +08:00
										 |  |  |         return; | 
					
						
							| 
									
										
										
										
											2008-05-08 14:29:10 +00:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2021-02-02 00:26:56 +08:00
										 |  |  | 
 | 
					
						
							|  |  |  |     PyObject *exc; | 
					
						
							| 
									
										
										
										
											2016-12-09 00:33:39 +01:00
										 |  |  |     exc = PyObject_CallFunction(JSONDecodeError, "zOn", msg, s, end); | 
					
						
							| 
									
										
										
										
											2021-02-02 00:26:56 +08:00
										 |  |  |     Py_DECREF(JSONDecodeError); | 
					
						
							| 
									
										
										
										
											2015-01-26 13:16:30 +02:00
										 |  |  |     if (exc) { | 
					
						
							|  |  |  |         PyErr_SetObject(JSONDecodeError, exc); | 
					
						
							|  |  |  |         Py_DECREF(exc); | 
					
						
							| 
									
										
										
										
											2008-10-16 21:17:24 +00:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2008-05-08 14:29:10 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-01-03 08:44:15 +02:00
										 |  |  | static void | 
					
						
							|  |  |  | raise_stop_iteration(Py_ssize_t idx) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     PyObject *value = PyLong_FromSsize_t(idx); | 
					
						
							|  |  |  |     if (value != NULL) { | 
					
						
							|  |  |  |         PyErr_SetObject(PyExc_StopIteration, value); | 
					
						
							|  |  |  |         Py_DECREF(value); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-05-08 14:29:10 +00:00
										 |  |  | static PyObject * | 
					
						
							| 
									
										
										
										
											2009-05-02 12:36:44 +00:00
										 |  |  | _build_rval_index_tuple(PyObject *rval, Py_ssize_t idx) { | 
					
						
							|  |  |  |     /* return (rval, idx) tuple, stealing reference to rval */ | 
					
						
							|  |  |  |     PyObject *tpl; | 
					
						
							|  |  |  |     PyObject *pyidx; | 
					
						
							|  |  |  |     /*
 | 
					
						
							|  |  |  |     steal a reference to rval, returns (rval, idx) | 
					
						
							|  |  |  |     */ | 
					
						
							|  |  |  |     if (rval == NULL) { | 
					
						
							|  |  |  |         return NULL; | 
					
						
							| 
									
										
										
										
											2008-07-19 22:26:35 +00:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2009-05-02 12:36:44 +00:00
										 |  |  |     pyidx = PyLong_FromSsize_t(idx); | 
					
						
							|  |  |  |     if (pyidx == NULL) { | 
					
						
							|  |  |  |         Py_DECREF(rval); | 
					
						
							|  |  |  |         return NULL; | 
					
						
							| 
									
										
										
										
											2008-05-08 14:29:10 +00:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2009-05-02 12:36:44 +00:00
										 |  |  |     tpl = PyTuple_New(2); | 
					
						
							|  |  |  |     if (tpl == NULL) { | 
					
						
							|  |  |  |         Py_DECREF(pyidx); | 
					
						
							|  |  |  |         Py_DECREF(rval); | 
					
						
							|  |  |  |         return NULL; | 
					
						
							| 
									
										
										
										
											2008-05-08 14:29:10 +00:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2009-05-02 12:36:44 +00:00
										 |  |  |     PyTuple_SET_ITEM(tpl, 0, rval); | 
					
						
							|  |  |  |     PyTuple_SET_ITEM(tpl, 1, pyidx); | 
					
						
							|  |  |  |     return tpl; | 
					
						
							| 
									
										
										
										
											2008-05-08 14:29:10 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static PyObject * | 
					
						
							| 
									
										
										
										
											2009-05-02 12:36:44 +00:00
										 |  |  | scanstring_unicode(PyObject *pystr, Py_ssize_t end, int strict, Py_ssize_t *next_end_ptr) | 
					
						
							| 
									
										
										
										
											2008-05-08 14:29:10 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2009-05-02 12:36:44 +00:00
										 |  |  |     /* Read the JSON string from PyUnicode pystr.
 | 
					
						
							|  |  |  |     end is the index of the first character after the quote. | 
					
						
							|  |  |  |     if strict is zero then literal control characters are allowed | 
					
						
							|  |  |  |     *next_end_ptr is a return-by-reference index of the character | 
					
						
							|  |  |  |         after the end quote | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     Return value is a new PyUnicode | 
					
						
							|  |  |  |     */ | 
					
						
							| 
									
										
										
										
											2010-09-04 20:16:53 +00:00
										 |  |  |     PyObject *rval = NULL; | 
					
						
							| 
									
										
										
										
											2011-09-28 07:41:54 +02:00
										 |  |  |     Py_ssize_t len; | 
					
						
							| 
									
										
										
										
											2008-05-08 14:29:10 +00:00
										 |  |  |     Py_ssize_t begin = end - 1; | 
					
						
							| 
									
										
										
										
											2011-02-22 20:15:44 +00:00
										 |  |  |     Py_ssize_t next /* = begin */; | 
					
						
							| 
									
										
										
										
											2011-09-28 07:41:54 +02:00
										 |  |  |     const void *buf; | 
					
						
							|  |  |  |     int kind; | 
					
						
							| 
									
										
										
										
											2010-09-04 20:16:53 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-09-28 07:41:54 +02:00
										 |  |  |     if (PyUnicode_READY(pystr) == -1) | 
					
						
							|  |  |  |         return 0; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-10-17 16:12:41 +09:00
										 |  |  |     _PyUnicodeWriter writer; | 
					
						
							|  |  |  |     _PyUnicodeWriter_Init(&writer); | 
					
						
							|  |  |  |     writer.overallocate = 1; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-09-28 07:41:54 +02:00
										 |  |  |     len = PyUnicode_GET_LENGTH(pystr); | 
					
						
							|  |  |  |     buf = PyUnicode_DATA(pystr); | 
					
						
							|  |  |  |     kind = PyUnicode_KIND(pystr); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-01-03 08:44:15 +02:00
										 |  |  |     if (end < 0 || len < end) { | 
					
						
							| 
									
										
										
										
											2008-07-19 22:26:35 +00:00
										 |  |  |         PyErr_SetString(PyExc_ValueError, "end is out of bounds"); | 
					
						
							|  |  |  |         goto bail; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2008-05-08 14:29:10 +00:00
										 |  |  |     while (1) { | 
					
						
							|  |  |  |         /* Find the end of the string or the next escape */ | 
					
						
							| 
									
										
										
										
											2019-08-08 17:57:10 +09:00
										 |  |  |         Py_UCS4 c; | 
					
						
							|  |  |  |         { | 
					
						
							|  |  |  |             // Use tight scope variable to help register allocation.
 | 
					
						
							|  |  |  |             Py_UCS4 d = 0; | 
					
						
							|  |  |  |             for (next = end; next < len; next++) { | 
					
						
							|  |  |  |                 d = PyUnicode_READ(kind, buf, next); | 
					
						
							|  |  |  |                 if (d == '"' || d == '\\') { | 
					
						
							|  |  |  |                     break; | 
					
						
							|  |  |  |                 } | 
					
						
							|  |  |  |                 if (d <= 0x1f && strict) { | 
					
						
							|  |  |  |                     raise_errmsg("Invalid control character at", pystr, next); | 
					
						
							|  |  |  |                     goto bail; | 
					
						
							|  |  |  |                 } | 
					
						
							| 
									
										
										
										
											2008-05-08 14:29:10 +00:00
										 |  |  |             } | 
					
						
							| 
									
										
										
										
											2019-08-08 17:57:10 +09:00
										 |  |  |             c = d; | 
					
						
							| 
									
										
										
										
											2008-05-08 14:29:10 +00:00
										 |  |  |         } | 
					
						
							| 
									
										
										
										
											2019-10-17 16:12:41 +09:00
										 |  |  | 
 | 
					
						
							|  |  |  |         if (c == '"') { | 
					
						
							|  |  |  |             // Fast path for simple case.
 | 
					
						
							|  |  |  |             if (writer.buffer == NULL) { | 
					
						
							|  |  |  |                 PyObject *ret = PyUnicode_Substring(pystr, end, next); | 
					
						
							|  |  |  |                 if (ret == NULL) { | 
					
						
							|  |  |  |                     goto bail; | 
					
						
							|  |  |  |                 } | 
					
						
							|  |  |  |                 *next_end_ptr = next + 1;; | 
					
						
							|  |  |  |                 return ret; | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         else if (c != '\\') { | 
					
						
							| 
									
										
										
										
											2008-05-08 14:29:10 +00:00
										 |  |  |             raise_errmsg("Unterminated string starting at", pystr, begin); | 
					
						
							|  |  |  |             goto bail; | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											2019-10-17 16:12:41 +09:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-05-08 14:29:10 +00:00
										 |  |  |         /* Pick up this chunk if it's not zero length */ | 
					
						
							|  |  |  |         if (next != end) { | 
					
						
							| 
									
										
										
										
											2019-10-17 16:12:41 +09:00
										 |  |  |             if (_PyUnicodeWriter_WriteSubstring(&writer, pystr, end, next) < 0) { | 
					
						
							| 
									
										
										
										
											2008-05-08 14:29:10 +00:00
										 |  |  |                 goto bail; | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         next++; | 
					
						
							|  |  |  |         if (c == '"') { | 
					
						
							|  |  |  |             end = next; | 
					
						
							|  |  |  |             break; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         if (next == len) { | 
					
						
							|  |  |  |             raise_errmsg("Unterminated string starting at", pystr, begin); | 
					
						
							|  |  |  |             goto bail; | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											2011-09-28 07:41:54 +02:00
										 |  |  |         c = PyUnicode_READ(kind, buf, next); | 
					
						
							| 
									
										
										
										
											2008-05-08 14:29:10 +00:00
										 |  |  |         if (c != 'u') { | 
					
						
							|  |  |  |             /* Non-unicode backslash escapes */ | 
					
						
							|  |  |  |             end = next + 1; | 
					
						
							|  |  |  |             switch (c) { | 
					
						
							|  |  |  |                 case '"': break; | 
					
						
							|  |  |  |                 case '\\': break; | 
					
						
							|  |  |  |                 case '/': break; | 
					
						
							|  |  |  |                 case 'b': c = '\b'; break; | 
					
						
							|  |  |  |                 case 'f': c = '\f'; break; | 
					
						
							|  |  |  |                 case 'n': c = '\n'; break; | 
					
						
							|  |  |  |                 case 'r': c = '\r'; break; | 
					
						
							|  |  |  |                 case 't': c = '\t'; break; | 
					
						
							|  |  |  |                 default: c = 0; | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |             if (c == 0) { | 
					
						
							|  |  |  |                 raise_errmsg("Invalid \\escape", pystr, end - 2); | 
					
						
							|  |  |  |                 goto bail; | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         else { | 
					
						
							|  |  |  |             c = 0; | 
					
						
							|  |  |  |             next++; | 
					
						
							|  |  |  |             end = next + 4; | 
					
						
							|  |  |  |             if (end >= len) { | 
					
						
							|  |  |  |                 raise_errmsg("Invalid \\uXXXX escape", pystr, next - 1); | 
					
						
							|  |  |  |                 goto bail; | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |             /* Decode 4 hex digits */ | 
					
						
							|  |  |  |             for (; next < end; next++) { | 
					
						
							| 
									
										
										
										
											2011-09-28 07:41:54 +02:00
										 |  |  |                 Py_UCS4 digit = PyUnicode_READ(kind, buf, next); | 
					
						
							| 
									
										
										
										
											2009-05-02 12:36:44 +00:00
										 |  |  |                 c <<= 4; | 
					
						
							| 
									
										
										
										
											2008-05-08 14:29:10 +00:00
										 |  |  |                 switch (digit) { | 
					
						
							|  |  |  |                     case '0': case '1': case '2': case '3': case '4': | 
					
						
							|  |  |  |                     case '5': case '6': case '7': case '8': case '9': | 
					
						
							| 
									
										
										
										
											2009-05-02 12:36:44 +00:00
										 |  |  |                         c |= (digit - '0'); break; | 
					
						
							| 
									
										
										
										
											2008-05-08 14:29:10 +00:00
										 |  |  |                     case 'a': case 'b': case 'c': case 'd': case 'e': | 
					
						
							|  |  |  |                     case 'f': | 
					
						
							| 
									
										
										
										
											2009-05-02 12:36:44 +00:00
										 |  |  |                         c |= (digit - 'a' + 10); break; | 
					
						
							| 
									
										
										
										
											2008-05-08 14:29:10 +00:00
										 |  |  |                     case 'A': case 'B': case 'C': case 'D': case 'E': | 
					
						
							|  |  |  |                     case 'F': | 
					
						
							| 
									
										
										
										
											2009-05-02 12:36:44 +00:00
										 |  |  |                         c |= (digit - 'A' + 10); break; | 
					
						
							| 
									
										
										
										
											2008-05-08 14:29:10 +00:00
										 |  |  |                     default: | 
					
						
							|  |  |  |                         raise_errmsg("Invalid \\uXXXX escape", pystr, end - 5); | 
					
						
							|  |  |  |                         goto bail; | 
					
						
							|  |  |  |                 } | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |             /* Surrogate pair */ | 
					
						
							| 
									
										
										
										
											2013-11-26 21:25:28 +02:00
										 |  |  |             if (Py_UNICODE_IS_HIGH_SURROGATE(c) && end + 6 < len && | 
					
						
							|  |  |  |                 PyUnicode_READ(kind, buf, next++) == '\\' && | 
					
						
							|  |  |  |                 PyUnicode_READ(kind, buf, next++) == 'u') { | 
					
						
							| 
									
										
										
										
											2011-09-28 07:41:54 +02:00
										 |  |  |                 Py_UCS4 c2 = 0; | 
					
						
							| 
									
										
										
										
											2008-05-08 14:29:10 +00:00
										 |  |  |                 end += 6; | 
					
						
							|  |  |  |                 /* Decode 4 hex digits */ | 
					
						
							|  |  |  |                 for (; next < end; next++) { | 
					
						
							| 
									
										
										
										
											2011-09-28 07:41:54 +02:00
										 |  |  |                     Py_UCS4 digit = PyUnicode_READ(kind, buf, next); | 
					
						
							| 
									
										
										
										
											2010-10-09 15:24:28 +00:00
										 |  |  |                     c2 <<= 4; | 
					
						
							| 
									
										
										
										
											2008-05-08 14:29:10 +00:00
										 |  |  |                     switch (digit) { | 
					
						
							|  |  |  |                         case '0': case '1': case '2': case '3': case '4': | 
					
						
							|  |  |  |                         case '5': case '6': case '7': case '8': case '9': | 
					
						
							| 
									
										
										
										
											2009-05-02 12:36:44 +00:00
										 |  |  |                             c2 |= (digit - '0'); break; | 
					
						
							| 
									
										
										
										
											2008-05-08 14:29:10 +00:00
										 |  |  |                         case 'a': case 'b': case 'c': case 'd': case 'e': | 
					
						
							|  |  |  |                         case 'f': | 
					
						
							| 
									
										
										
										
											2009-05-02 12:36:44 +00:00
										 |  |  |                             c2 |= (digit - 'a' + 10); break; | 
					
						
							| 
									
										
										
										
											2008-05-08 14:29:10 +00:00
										 |  |  |                         case 'A': case 'B': case 'C': case 'D': case 'E': | 
					
						
							|  |  |  |                         case 'F': | 
					
						
							| 
									
										
										
										
											2009-05-02 12:36:44 +00:00
										 |  |  |                             c2 |= (digit - 'A' + 10); break; | 
					
						
							| 
									
										
										
										
											2008-05-08 14:29:10 +00:00
										 |  |  |                         default: | 
					
						
							|  |  |  |                             raise_errmsg("Invalid \\uXXXX escape", pystr, end - 5); | 
					
						
							|  |  |  |                             goto bail; | 
					
						
							|  |  |  |                     } | 
					
						
							|  |  |  |                 } | 
					
						
							| 
									
										
										
										
											2013-11-26 21:25:28 +02:00
										 |  |  |                 if (Py_UNICODE_IS_LOW_SURROGATE(c2)) | 
					
						
							|  |  |  |                     c = Py_UNICODE_JOIN_SURROGATES(c, c2); | 
					
						
							|  |  |  |                 else | 
					
						
							|  |  |  |                     end -= 6; | 
					
						
							| 
									
										
										
										
											2009-05-02 12:36:44 +00:00
										 |  |  |             } | 
					
						
							| 
									
										
										
										
											2008-05-08 14:29:10 +00:00
										 |  |  |         } | 
					
						
							| 
									
										
										
										
											2019-10-17 16:12:41 +09:00
										 |  |  |         if (_PyUnicodeWriter_WriteChar(&writer, c) < 0) { | 
					
						
							| 
									
										
										
										
											2008-05-08 14:29:10 +00:00
										 |  |  |             goto bail; | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											2010-09-04 20:16:53 +00:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-10-17 16:12:41 +09:00
										 |  |  |     rval = _PyUnicodeWriter_Finish(&writer); | 
					
						
							| 
									
										
										
										
											2009-05-02 12:36:44 +00:00
										 |  |  |     *next_end_ptr = end; | 
					
						
							|  |  |  |     return rval; | 
					
						
							| 
									
										
										
										
											2019-10-17 16:12:41 +09:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-05-08 14:29:10 +00:00
										 |  |  | bail: | 
					
						
							| 
									
										
										
										
											2009-05-02 12:36:44 +00:00
										 |  |  |     *next_end_ptr = -1; | 
					
						
							| 
									
										
										
										
											2019-10-17 16:12:41 +09:00
										 |  |  |     _PyUnicodeWriter_Dealloc(&writer); | 
					
						
							| 
									
										
										
										
											2008-05-08 14:29:10 +00:00
										 |  |  |     return NULL; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | PyDoc_STRVAR(pydoc_scanstring, | 
					
						
							| 
									
										
										
										
											2010-08-02 20:16:18 +00:00
										 |  |  |     "scanstring(string, end, strict=True) -> (string, end)\n" | 
					
						
							| 
									
										
										
										
											2009-05-02 12:36:44 +00:00
										 |  |  |     "\n" | 
					
						
							|  |  |  |     "Scan the string s for a JSON string. End is the index of the\n" | 
					
						
							|  |  |  |     "character in s after the quote that started the JSON string.\n" | 
					
						
							|  |  |  |     "Unescapes all valid JSON string escape sequences and raises ValueError\n" | 
					
						
							|  |  |  |     "on attempt to decode an invalid string. If strict is False then literal\n" | 
					
						
							|  |  |  |     "control characters are allowed in the string.\n" | 
					
						
							|  |  |  |     "\n" | 
					
						
							|  |  |  |     "Returns a tuple of the decoded string and the index of the character in s\n" | 
					
						
							|  |  |  |     "after the end quote." | 
					
						
							|  |  |  | ); | 
					
						
							| 
									
										
										
										
											2008-05-08 14:29:10 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | static PyObject * | 
					
						
							| 
									
										
										
										
											2019-11-05 11:44:28 +01:00
										 |  |  | py_scanstring(PyObject* Py_UNUSED(self), PyObject *args) | 
					
						
							| 
									
										
										
										
											2008-05-08 14:29:10 +00:00
										 |  |  | { | 
					
						
							|  |  |  |     PyObject *pystr; | 
					
						
							| 
									
										
										
										
											2009-05-02 12:36:44 +00:00
										 |  |  |     PyObject *rval; | 
					
						
							| 
									
										
										
										
											2008-05-08 14:29:10 +00:00
										 |  |  |     Py_ssize_t end; | 
					
						
							| 
									
										
										
										
											2009-05-02 12:36:44 +00:00
										 |  |  |     Py_ssize_t next_end = -1; | 
					
						
							|  |  |  |     int strict = 1; | 
					
						
							| 
									
										
										
										
											2012-12-01 19:34:16 +01:00
										 |  |  |     if (!PyArg_ParseTuple(args, "On|i:scanstring", &pystr, &end, &strict)) { | 
					
						
							| 
									
										
										
										
											2008-05-08 14:29:10 +00:00
										 |  |  |         return NULL; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2009-05-02 12:36:44 +00:00
										 |  |  |     if (PyUnicode_Check(pystr)) { | 
					
						
							|  |  |  |         rval = scanstring_unicode(pystr, end, strict, &next_end); | 
					
						
							| 
									
										
										
										
											2008-05-08 14:29:10 +00:00
										 |  |  |     } | 
					
						
							|  |  |  |     else { | 
					
						
							| 
									
										
										
										
											2010-05-09 15:52:27 +00:00
										 |  |  |         PyErr_Format(PyExc_TypeError, | 
					
						
							| 
									
										
										
										
											2010-08-02 20:16:18 +00:00
										 |  |  |                      "first argument must be a string, not %.80s", | 
					
						
							| 
									
										
										
										
											2008-05-08 14:29:10 +00:00
										 |  |  |                      Py_TYPE(pystr)->tp_name); | 
					
						
							|  |  |  |         return NULL; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2009-05-02 12:36:44 +00:00
										 |  |  |     return _build_rval_index_tuple(rval, next_end); | 
					
						
							| 
									
										
										
										
											2008-05-08 14:29:10 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | PyDoc_STRVAR(pydoc_encode_basestring_ascii, | 
					
						
							| 
									
										
										
										
											2010-08-02 20:16:18 +00:00
										 |  |  |     "encode_basestring_ascii(string) -> string\n" | 
					
						
							| 
									
										
										
										
											2009-05-02 12:36:44 +00:00
										 |  |  |     "\n" | 
					
						
							|  |  |  |     "Return an ASCII-only JSON representation of a Python string" | 
					
						
							|  |  |  | ); | 
					
						
							| 
									
										
										
										
											2008-05-08 14:29:10 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | static PyObject * | 
					
						
							| 
									
										
										
										
											2019-11-05 11:44:28 +01:00
										 |  |  | py_encode_basestring_ascii(PyObject* Py_UNUSED(self), PyObject *pystr) | 
					
						
							| 
									
										
										
										
											2008-05-08 14:29:10 +00:00
										 |  |  | { | 
					
						
							|  |  |  |     PyObject *rval; | 
					
						
							| 
									
										
										
										
											2009-05-02 12:36:44 +00:00
										 |  |  |     /* Return an ASCII-only JSON representation of a Python string */ | 
					
						
							| 
									
										
										
										
											2008-05-08 14:29:10 +00:00
										 |  |  |     /* METH_O */ | 
					
						
							| 
									
										
										
										
											2009-05-02 12:36:44 +00:00
										 |  |  |     if (PyUnicode_Check(pystr)) { | 
					
						
							| 
									
										
										
										
											2008-05-08 14:29:10 +00:00
										 |  |  |         rval = ascii_escape_unicode(pystr); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     else { | 
					
						
							| 
									
										
										
										
											2009-05-02 12:36:44 +00:00
										 |  |  |         PyErr_Format(PyExc_TypeError, | 
					
						
							|  |  |  |                      "first argument must be a string, not %.80s", | 
					
						
							| 
									
										
										
										
											2008-05-08 14:29:10 +00:00
										 |  |  |                      Py_TYPE(pystr)->tp_name); | 
					
						
							|  |  |  |         return NULL; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     return rval; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-01-11 16:41:01 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  | PyDoc_STRVAR(pydoc_encode_basestring, | 
					
						
							|  |  |  |     "encode_basestring(string) -> string\n" | 
					
						
							|  |  |  |     "\n" | 
					
						
							|  |  |  |     "Return a JSON representation of a Python string" | 
					
						
							|  |  |  | ); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static PyObject * | 
					
						
							| 
									
										
										
										
											2019-11-05 11:44:28 +01:00
										 |  |  | py_encode_basestring(PyObject* Py_UNUSED(self), PyObject *pystr) | 
					
						
							| 
									
										
										
										
											2015-01-11 16:41:01 +01:00
										 |  |  | { | 
					
						
							|  |  |  |     PyObject *rval; | 
					
						
							|  |  |  |     /* Return a JSON representation of a Python string */ | 
					
						
							|  |  |  |     /* METH_O */ | 
					
						
							|  |  |  |     if (PyUnicode_Check(pystr)) { | 
					
						
							|  |  |  |         rval = escape_unicode(pystr); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     else { | 
					
						
							|  |  |  |         PyErr_Format(PyExc_TypeError, | 
					
						
							|  |  |  |                      "first argument must be a string, not %.80s", | 
					
						
							|  |  |  |                      Py_TYPE(pystr)->tp_name); | 
					
						
							|  |  |  |         return NULL; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     return rval; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-05-02 12:36:44 +00:00
										 |  |  | static void | 
					
						
							|  |  |  | scanner_dealloc(PyObject *self) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2020-03-27 19:59:59 +09:00
										 |  |  |     PyTypeObject *tp = Py_TYPE(self); | 
					
						
							| 
									
										
										
										
											2017-08-24 14:55:17 +09:00
										 |  |  |     /* bpo-31095: UnTrack is needed before calling any callbacks */ | 
					
						
							|  |  |  |     PyObject_GC_UnTrack(self); | 
					
						
							| 
									
										
										
										
											2020-03-27 19:59:59 +09:00
										 |  |  |     scanner_clear((PyScannerObject *)self); | 
					
						
							|  |  |  |     tp->tp_free(self); | 
					
						
							|  |  |  |     Py_DECREF(tp); | 
					
						
							| 
									
										
										
										
											2009-05-02 12:36:44 +00:00
										 |  |  | } | 
					
						
							| 
									
										
										
										
											2008-05-08 14:29:10 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-05-02 12:36:44 +00:00
										 |  |  | static int | 
					
						
							| 
									
										
										
										
											2020-03-27 19:59:59 +09:00
										 |  |  | scanner_traverse(PyScannerObject *self, visitproc visit, void *arg) | 
					
						
							| 
									
										
										
										
											2009-05-02 12:36:44 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2020-05-27 10:03:38 +01:00
										 |  |  |     Py_VISIT(Py_TYPE(self)); | 
					
						
							| 
									
										
										
										
											2020-03-27 19:59:59 +09:00
										 |  |  |     Py_VISIT(self->object_hook); | 
					
						
							|  |  |  |     Py_VISIT(self->object_pairs_hook); | 
					
						
							|  |  |  |     Py_VISIT(self->parse_float); | 
					
						
							|  |  |  |     Py_VISIT(self->parse_int); | 
					
						
							|  |  |  |     Py_VISIT(self->parse_constant); | 
					
						
							| 
									
										
										
										
											2020-04-05 03:24:16 +08:00
										 |  |  |     Py_VISIT(self->memo); | 
					
						
							| 
									
										
										
										
											2009-05-02 12:36:44 +00:00
										 |  |  |     return 0; | 
					
						
							|  |  |  | } | 
					
						
							| 
									
										
										
										
											2008-06-11 05:26:20 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-05-02 12:36:44 +00:00
										 |  |  | static int | 
					
						
							| 
									
										
										
										
											2020-03-27 19:59:59 +09:00
										 |  |  | scanner_clear(PyScannerObject *self) | 
					
						
							| 
									
										
										
										
											2008-05-08 14:29:10 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2020-03-27 19:59:59 +09:00
										 |  |  |     Py_CLEAR(self->object_hook); | 
					
						
							|  |  |  |     Py_CLEAR(self->object_pairs_hook); | 
					
						
							|  |  |  |     Py_CLEAR(self->parse_float); | 
					
						
							|  |  |  |     Py_CLEAR(self->parse_int); | 
					
						
							|  |  |  |     Py_CLEAR(self->parse_constant); | 
					
						
							|  |  |  |     Py_CLEAR(self->memo); | 
					
						
							| 
									
										
										
										
											2009-05-02 12:36:44 +00:00
										 |  |  |     return 0; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static PyObject * | 
					
						
							| 
									
										
										
										
											2017-05-28 15:31:49 +03:00
										 |  |  | _parse_object_unicode(PyScannerObject *s, PyObject *pystr, Py_ssize_t idx, Py_ssize_t *next_idx_ptr) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2009-05-02 12:36:44 +00:00
										 |  |  |     /* Read a JSON object from PyUnicode pystr.
 | 
					
						
							|  |  |  |     idx is the index of the first character after the opening curly brace. | 
					
						
							|  |  |  |     *next_idx_ptr is a return-by-reference index to the first character after | 
					
						
							|  |  |  |         the closing curly brace. | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     Returns a new PyObject (usually a dict, but object_hook can change that) | 
					
						
							|  |  |  |     */ | 
					
						
							| 
									
										
										
										
											2020-04-11 10:48:40 +03:00
										 |  |  |     const void *str; | 
					
						
							| 
									
										
										
										
											2011-09-28 07:41:54 +02:00
										 |  |  |     int kind; | 
					
						
							|  |  |  |     Py_ssize_t end_idx; | 
					
						
							| 
									
										
										
										
											2009-05-02 12:36:44 +00:00
										 |  |  |     PyObject *val = NULL; | 
					
						
							| 
									
										
										
										
											2010-09-04 20:16:53 +00:00
										 |  |  |     PyObject *rval = NULL; | 
					
						
							| 
									
										
										
										
											2009-05-02 12:36:44 +00:00
										 |  |  |     PyObject *key = NULL; | 
					
						
							| 
									
										
										
										
											2010-09-04 20:16:53 +00:00
										 |  |  |     int has_pairs_hook = (s->object_pairs_hook != Py_None); | 
					
						
							| 
									
										
										
										
											2009-05-02 12:36:44 +00:00
										 |  |  |     Py_ssize_t next_idx; | 
					
						
							| 
									
										
										
										
											2010-09-04 20:16:53 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-09-28 07:41:54 +02:00
										 |  |  |     if (PyUnicode_READY(pystr) == -1) | 
					
						
							|  |  |  |         return NULL; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     str = PyUnicode_DATA(pystr); | 
					
						
							|  |  |  |     kind = PyUnicode_KIND(pystr); | 
					
						
							|  |  |  |     end_idx = PyUnicode_GET_LENGTH(pystr) - 1; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-09-04 20:16:53 +00:00
										 |  |  |     if (has_pairs_hook) | 
					
						
							|  |  |  |         rval = PyList_New(0); | 
					
						
							|  |  |  |     else | 
					
						
							|  |  |  |         rval = PyDict_New(); | 
					
						
							| 
									
										
										
										
											2009-05-02 12:36:44 +00:00
										 |  |  |     if (rval == NULL) | 
					
						
							|  |  |  |         return NULL; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     /* skip whitespace after { */ | 
					
						
							| 
									
										
										
										
											2011-09-28 07:41:54 +02:00
										 |  |  |     while (idx <= end_idx && IS_WHITESPACE(PyUnicode_READ(kind,str, idx))) idx++; | 
					
						
							| 
									
										
										
										
											2009-05-02 12:36:44 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |     /* only loop if the object is non-empty */ | 
					
						
							| 
									
										
										
										
											2013-01-03 08:44:15 +02:00
										 |  |  |     if (idx > end_idx || PyUnicode_READ(kind, str, idx) != '}') { | 
					
						
							|  |  |  |         while (1) { | 
					
						
							| 
									
										
										
										
											2010-09-04 20:16:53 +00:00
										 |  |  |             PyObject *memokey; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-05-02 12:36:44 +00:00
										 |  |  |             /* read key */ | 
					
						
							| 
									
										
										
										
											2013-01-03 08:44:15 +02:00
										 |  |  |             if (idx > end_idx || PyUnicode_READ(kind, str, idx) != '"') { | 
					
						
							| 
									
										
										
										
											2012-06-29 01:58:26 +02:00
										 |  |  |                 raise_errmsg("Expecting property name enclosed in double quotes", pystr, idx); | 
					
						
							| 
									
										
										
										
											2009-05-02 12:36:44 +00:00
										 |  |  |                 goto bail; | 
					
						
							|  |  |  |             } | 
					
						
							| 
									
										
										
										
											2017-05-28 15:31:49 +03:00
										 |  |  |             key = scanstring_unicode(pystr, idx + 1, s->strict, &next_idx); | 
					
						
							| 
									
										
										
										
											2009-05-02 12:36:44 +00:00
										 |  |  |             if (key == NULL) | 
					
						
							|  |  |  |                 goto bail; | 
					
						
							| 
									
										
										
										
											2019-08-08 17:57:10 +09:00
										 |  |  |             memokey = PyDict_SetDefault(s->memo, key, key); | 
					
						
							|  |  |  |             if (memokey == NULL) { | 
					
						
							| 
									
										
										
										
											2019-02-25 17:59:46 +02:00
										 |  |  |                 goto bail; | 
					
						
							|  |  |  |             } | 
					
						
							| 
									
										
										
										
											2019-08-08 17:57:10 +09:00
										 |  |  |             Py_INCREF(memokey); | 
					
						
							|  |  |  |             Py_DECREF(key); | 
					
						
							|  |  |  |             key = memokey; | 
					
						
							| 
									
										
										
										
											2009-05-02 12:36:44 +00:00
										 |  |  |             idx = next_idx; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             /* skip whitespace between key and : delimiter, read :, skip whitespace */ | 
					
						
							| 
									
										
										
										
											2011-09-28 07:41:54 +02:00
										 |  |  |             while (idx <= end_idx && IS_WHITESPACE(PyUnicode_READ(kind, str, idx))) idx++; | 
					
						
							|  |  |  |             if (idx > end_idx || PyUnicode_READ(kind, str, idx) != ':') { | 
					
						
							| 
									
										
										
										
											2012-06-29 01:58:26 +02:00
										 |  |  |                 raise_errmsg("Expecting ':' delimiter", pystr, idx); | 
					
						
							| 
									
										
										
										
											2009-05-02 12:36:44 +00:00
										 |  |  |                 goto bail; | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |             idx++; | 
					
						
							| 
									
										
										
										
											2011-09-28 07:41:54 +02:00
										 |  |  |             while (idx <= end_idx && IS_WHITESPACE(PyUnicode_READ(kind, str, idx))) idx++; | 
					
						
							| 
									
										
										
										
											2009-05-02 12:36:44 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |             /* read any JSON term */ | 
					
						
							|  |  |  |             val = scan_once_unicode(s, pystr, idx, &next_idx); | 
					
						
							|  |  |  |             if (val == NULL) | 
					
						
							|  |  |  |                 goto bail; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-09-04 20:16:53 +00:00
										 |  |  |             if (has_pairs_hook) { | 
					
						
							|  |  |  |                 PyObject *item = PyTuple_Pack(2, key, val); | 
					
						
							|  |  |  |                 if (item == NULL) | 
					
						
							| 
									
										
										
										
											2009-05-02 12:36:44 +00:00
										 |  |  |                     goto bail; | 
					
						
							| 
									
										
										
										
											2010-09-04 20:16:53 +00:00
										 |  |  |                 Py_CLEAR(key); | 
					
						
							|  |  |  |                 Py_CLEAR(val); | 
					
						
							|  |  |  |                 if (PyList_Append(rval, item) == -1) { | 
					
						
							|  |  |  |                     Py_DECREF(item); | 
					
						
							| 
									
										
										
										
											2009-05-02 12:36:44 +00:00
										 |  |  |                     goto bail; | 
					
						
							|  |  |  |                 } | 
					
						
							| 
									
										
										
										
											2010-09-04 20:16:53 +00:00
										 |  |  |                 Py_DECREF(item); | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |             else { | 
					
						
							|  |  |  |                 if (PyDict_SetItem(rval, key, val) < 0) | 
					
						
							|  |  |  |                     goto bail; | 
					
						
							|  |  |  |                 Py_CLEAR(key); | 
					
						
							|  |  |  |                 Py_CLEAR(val); | 
					
						
							| 
									
										
										
										
											2009-05-02 12:36:44 +00:00
										 |  |  |             } | 
					
						
							|  |  |  |             idx = next_idx; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             /* skip whitespace before } or , */ | 
					
						
							| 
									
										
										
										
											2011-09-28 07:41:54 +02:00
										 |  |  |             while (idx <= end_idx && IS_WHITESPACE(PyUnicode_READ(kind, str, idx))) idx++; | 
					
						
							| 
									
										
										
										
											2009-05-02 12:36:44 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |             /* bail if the object is closed or we didn't get the , delimiter */ | 
					
						
							| 
									
										
										
										
											2013-01-03 08:44:15 +02:00
										 |  |  |             if (idx <= end_idx && PyUnicode_READ(kind, str, idx) == '}') | 
					
						
							| 
									
										
										
										
											2009-05-02 12:36:44 +00:00
										 |  |  |                 break; | 
					
						
							| 
									
										
										
										
											2013-01-03 08:44:15 +02:00
										 |  |  |             if (idx > end_idx || PyUnicode_READ(kind, str, idx) != ',') { | 
					
						
							| 
									
										
										
										
											2012-06-29 01:58:26 +02:00
										 |  |  |                 raise_errmsg("Expecting ',' delimiter", pystr, idx); | 
					
						
							| 
									
										
										
										
											2009-05-02 12:36:44 +00:00
										 |  |  |                 goto bail; | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |             idx++; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             /* skip whitespace after , delimiter */ | 
					
						
							| 
									
										
										
										
											2011-09-28 07:41:54 +02:00
										 |  |  |             while (idx <= end_idx && IS_WHITESPACE(PyUnicode_READ(kind, str, idx))) idx++; | 
					
						
							| 
									
										
										
										
											2009-05-02 12:36:44 +00:00
										 |  |  |         } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     *next_idx_ptr = idx + 1; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-09-04 20:16:53 +00:00
										 |  |  |     if (has_pairs_hook) { | 
					
						
							| 
									
										
										
										
											2020-02-11 17:46:57 +01:00
										 |  |  |         val = PyObject_CallOneArg(s->object_pairs_hook, rval); | 
					
						
							| 
									
										
										
										
											2009-05-02 12:36:44 +00:00
										 |  |  |         Py_DECREF(rval); | 
					
						
							|  |  |  |         return val; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     /* if object_hook is not None: rval = object_hook(rval) */ | 
					
						
							|  |  |  |     if (s->object_hook != Py_None) { | 
					
						
							| 
									
										
										
										
											2020-02-11 17:46:57 +01:00
										 |  |  |         val = PyObject_CallOneArg(s->object_hook, rval); | 
					
						
							| 
									
										
										
										
											2009-05-02 12:36:44 +00:00
										 |  |  |         Py_DECREF(rval); | 
					
						
							| 
									
										
										
										
											2010-09-04 20:16:53 +00:00
										 |  |  |         return val; | 
					
						
							| 
									
										
										
										
											2009-05-02 12:36:44 +00:00
										 |  |  |     } | 
					
						
							|  |  |  |     return rval; | 
					
						
							|  |  |  | bail: | 
					
						
							|  |  |  |     Py_XDECREF(key); | 
					
						
							|  |  |  |     Py_XDECREF(val); | 
					
						
							| 
									
										
										
										
											2010-09-04 20:16:53 +00:00
										 |  |  |     Py_XDECREF(rval); | 
					
						
							| 
									
										
										
										
											2009-05-02 12:36:44 +00:00
										 |  |  |     return NULL; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static PyObject * | 
					
						
							|  |  |  | _parse_array_unicode(PyScannerObject *s, PyObject *pystr, Py_ssize_t idx, Py_ssize_t *next_idx_ptr) { | 
					
						
							| 
									
										
										
										
											2015-02-17 10:14:30 +02:00
										 |  |  |     /* Read a JSON array from PyUnicode pystr.
 | 
					
						
							| 
									
										
										
										
											2009-05-02 12:36:44 +00:00
										 |  |  |     idx is the index of the first character after the opening brace. | 
					
						
							|  |  |  |     *next_idx_ptr is a return-by-reference index to the first character after | 
					
						
							|  |  |  |         the closing brace. | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     Returns a new PyList | 
					
						
							|  |  |  |     */ | 
					
						
							| 
									
										
										
										
											2020-04-11 10:48:40 +03:00
										 |  |  |     const void *str; | 
					
						
							| 
									
										
										
										
											2011-09-28 07:41:54 +02:00
										 |  |  |     int kind; | 
					
						
							|  |  |  |     Py_ssize_t end_idx; | 
					
						
							| 
									
										
										
										
											2009-05-02 12:36:44 +00:00
										 |  |  |     PyObject *val = NULL; | 
					
						
							| 
									
										
										
										
											2017-01-03 11:17:44 +02:00
										 |  |  |     PyObject *rval; | 
					
						
							| 
									
										
										
										
											2009-05-02 12:36:44 +00:00
										 |  |  |     Py_ssize_t next_idx; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-09-28 07:41:54 +02:00
										 |  |  |     if (PyUnicode_READY(pystr) == -1) | 
					
						
							|  |  |  |         return NULL; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-01-03 11:17:44 +02:00
										 |  |  |     rval = PyList_New(0); | 
					
						
							|  |  |  |     if (rval == NULL) | 
					
						
							|  |  |  |         return NULL; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-09-28 07:41:54 +02:00
										 |  |  |     str = PyUnicode_DATA(pystr); | 
					
						
							|  |  |  |     kind = PyUnicode_KIND(pystr); | 
					
						
							|  |  |  |     end_idx = PyUnicode_GET_LENGTH(pystr) - 1; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-05-02 12:36:44 +00:00
										 |  |  |     /* skip whitespace after [ */ | 
					
						
							| 
									
										
										
										
											2011-09-28 07:41:54 +02:00
										 |  |  |     while (idx <= end_idx && IS_WHITESPACE(PyUnicode_READ(kind, str, idx))) idx++; | 
					
						
							| 
									
										
										
										
											2009-05-02 12:36:44 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |     /* only loop if the array is non-empty */ | 
					
						
							| 
									
										
										
										
											2013-01-03 08:44:15 +02:00
										 |  |  |     if (idx > end_idx || PyUnicode_READ(kind, str, idx) != ']') { | 
					
						
							|  |  |  |         while (1) { | 
					
						
							| 
									
										
										
										
											2009-05-02 12:36:44 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |             /* read any JSON term  */ | 
					
						
							|  |  |  |             val = scan_once_unicode(s, pystr, idx, &next_idx); | 
					
						
							|  |  |  |             if (val == NULL) | 
					
						
							|  |  |  |                 goto bail; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             if (PyList_Append(rval, val) == -1) | 
					
						
							|  |  |  |                 goto bail; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             Py_CLEAR(val); | 
					
						
							|  |  |  |             idx = next_idx; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             /* skip whitespace between term and , */ | 
					
						
							| 
									
										
										
										
											2011-09-28 07:41:54 +02:00
										 |  |  |             while (idx <= end_idx && IS_WHITESPACE(PyUnicode_READ(kind, str, idx))) idx++; | 
					
						
							| 
									
										
										
										
											2009-05-02 12:36:44 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |             /* bail if the array is closed or we didn't get the , delimiter */ | 
					
						
							| 
									
										
										
										
											2013-01-03 08:44:15 +02:00
										 |  |  |             if (idx <= end_idx && PyUnicode_READ(kind, str, idx) == ']') | 
					
						
							| 
									
										
										
										
											2009-05-02 12:36:44 +00:00
										 |  |  |                 break; | 
					
						
							| 
									
										
										
										
											2013-01-03 08:44:15 +02:00
										 |  |  |             if (idx > end_idx || PyUnicode_READ(kind, str, idx) != ',') { | 
					
						
							| 
									
										
										
										
											2012-06-29 01:58:26 +02:00
										 |  |  |                 raise_errmsg("Expecting ',' delimiter", pystr, idx); | 
					
						
							| 
									
										
										
										
											2009-05-02 12:36:44 +00:00
										 |  |  |                 goto bail; | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |             idx++; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             /* skip whitespace after , */ | 
					
						
							| 
									
										
										
										
											2011-09-28 07:41:54 +02:00
										 |  |  |             while (idx <= end_idx && IS_WHITESPACE(PyUnicode_READ(kind, str, idx))) idx++; | 
					
						
							| 
									
										
										
										
											2009-05-02 12:36:44 +00:00
										 |  |  |         } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-09-28 07:41:54 +02:00
										 |  |  |     /* verify that idx < end_idx, PyUnicode_READ(kind, str, idx) should be ']' */ | 
					
						
							|  |  |  |     if (idx > end_idx || PyUnicode_READ(kind, str, idx) != ']') { | 
					
						
							| 
									
										
										
										
											2013-01-03 08:44:15 +02:00
										 |  |  |         raise_errmsg("Expecting value", pystr, end_idx); | 
					
						
							| 
									
										
										
										
											2009-05-02 12:36:44 +00:00
										 |  |  |         goto bail; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     *next_idx_ptr = idx + 1; | 
					
						
							|  |  |  |     return rval; | 
					
						
							|  |  |  | bail: | 
					
						
							|  |  |  |     Py_XDECREF(val); | 
					
						
							|  |  |  |     Py_DECREF(rval); | 
					
						
							|  |  |  |     return NULL; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static PyObject * | 
					
						
							| 
									
										
										
										
											2015-02-17 10:14:30 +02:00
										 |  |  | _parse_constant(PyScannerObject *s, const char *constant, Py_ssize_t idx, Py_ssize_t *next_idx_ptr) { | 
					
						
							|  |  |  |     /* Read a JSON constant.
 | 
					
						
							| 
									
										
										
										
											2009-05-02 12:36:44 +00:00
										 |  |  |     constant is the constant string that was found | 
					
						
							|  |  |  |         ("NaN", "Infinity", "-Infinity"). | 
					
						
							|  |  |  |     idx is the index of the first character of the constant | 
					
						
							|  |  |  |     *next_idx_ptr is a return-by-reference index to the first character after | 
					
						
							|  |  |  |         the constant. | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     Returns the result of parse_constant | 
					
						
							|  |  |  |     */ | 
					
						
							|  |  |  |     PyObject *cstr; | 
					
						
							|  |  |  |     PyObject *rval; | 
					
						
							|  |  |  |     /* constant is "NaN", "Infinity", or "-Infinity" */ | 
					
						
							|  |  |  |     cstr = PyUnicode_InternFromString(constant); | 
					
						
							|  |  |  |     if (cstr == NULL) | 
					
						
							|  |  |  |         return NULL; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     /* rval = parse_constant(constant) */ | 
					
						
							| 
									
										
										
										
											2020-02-11 17:46:57 +01:00
										 |  |  |     rval = PyObject_CallOneArg(s->parse_constant, cstr); | 
					
						
							| 
									
										
										
										
											2011-10-11 22:11:42 +02:00
										 |  |  |     idx += PyUnicode_GET_LENGTH(cstr); | 
					
						
							| 
									
										
										
										
											2009-05-02 12:36:44 +00:00
										 |  |  |     Py_DECREF(cstr); | 
					
						
							|  |  |  |     *next_idx_ptr = idx; | 
					
						
							|  |  |  |     return rval; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static PyObject * | 
					
						
							|  |  |  | _match_number_unicode(PyScannerObject *s, PyObject *pystr, Py_ssize_t start, Py_ssize_t *next_idx_ptr) { | 
					
						
							|  |  |  |     /* Read a JSON number from PyUnicode pystr.
 | 
					
						
							|  |  |  |     idx is the index of the first character of the number | 
					
						
							|  |  |  |     *next_idx_ptr is a return-by-reference index to the first character after | 
					
						
							|  |  |  |         the number. | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     Returns a new PyObject representation of that number: | 
					
						
							| 
									
										
										
										
											2015-02-17 10:14:30 +02:00
										 |  |  |         PyLong, or PyFloat. | 
					
						
							| 
									
										
										
										
											2009-05-02 12:36:44 +00:00
										 |  |  |         May return other types if parse_int or parse_float are set | 
					
						
							|  |  |  |     */ | 
					
						
							| 
									
										
										
										
											2020-04-11 10:48:40 +03:00
										 |  |  |     const void *str; | 
					
						
							| 
									
										
										
										
											2011-09-28 07:41:54 +02:00
										 |  |  |     int kind; | 
					
						
							|  |  |  |     Py_ssize_t end_idx; | 
					
						
							| 
									
										
										
										
											2009-05-02 12:36:44 +00:00
										 |  |  |     Py_ssize_t idx = start; | 
					
						
							|  |  |  |     int is_float = 0; | 
					
						
							|  |  |  |     PyObject *rval; | 
					
						
							| 
									
										
										
										
											2011-04-25 19:16:06 +02:00
										 |  |  |     PyObject *numstr = NULL; | 
					
						
							|  |  |  |     PyObject *custom_func; | 
					
						
							| 
									
										
										
										
											2009-05-02 12:36:44 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-09-28 07:41:54 +02:00
										 |  |  |     if (PyUnicode_READY(pystr) == -1) | 
					
						
							|  |  |  |         return NULL; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     str = PyUnicode_DATA(pystr); | 
					
						
							|  |  |  |     kind = PyUnicode_KIND(pystr); | 
					
						
							|  |  |  |     end_idx = PyUnicode_GET_LENGTH(pystr) - 1; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-05-02 12:36:44 +00:00
										 |  |  |     /* read a sign if it's there, make sure it's not the end of the string */ | 
					
						
							| 
									
										
										
										
											2011-09-28 07:41:54 +02:00
										 |  |  |     if (PyUnicode_READ(kind, str, idx) == '-') { | 
					
						
							| 
									
										
										
										
											2009-05-02 12:36:44 +00:00
										 |  |  |         idx++; | 
					
						
							|  |  |  |         if (idx > end_idx) { | 
					
						
							| 
									
										
										
										
											2013-01-03 08:44:15 +02:00
										 |  |  |             raise_stop_iteration(start); | 
					
						
							| 
									
										
										
										
											2009-05-02 12:36:44 +00:00
										 |  |  |             return NULL; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     /* read as many integer digits as we find as long as it doesn't start with 0 */ | 
					
						
							| 
									
										
										
										
											2011-09-28 07:41:54 +02:00
										 |  |  |     if (PyUnicode_READ(kind, str, idx) >= '1' && PyUnicode_READ(kind, str, idx) <= '9') { | 
					
						
							| 
									
										
										
										
											2009-05-02 12:36:44 +00:00
										 |  |  |         idx++; | 
					
						
							| 
									
										
										
										
											2011-09-28 07:41:54 +02:00
										 |  |  |         while (idx <= end_idx && PyUnicode_READ(kind, str, idx) >= '0' && PyUnicode_READ(kind, str, idx) <= '9') idx++; | 
					
						
							| 
									
										
										
										
											2009-05-02 12:36:44 +00:00
										 |  |  |     } | 
					
						
							|  |  |  |     /* if it starts with 0 we only expect one integer digit */ | 
					
						
							| 
									
										
										
										
											2011-09-28 07:41:54 +02:00
										 |  |  |     else if (PyUnicode_READ(kind, str, idx) == '0') { | 
					
						
							| 
									
										
										
										
											2009-05-02 12:36:44 +00:00
										 |  |  |         idx++; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     /* no integer digits, error */ | 
					
						
							|  |  |  |     else { | 
					
						
							| 
									
										
										
										
											2013-01-03 08:44:15 +02:00
										 |  |  |         raise_stop_iteration(start); | 
					
						
							| 
									
										
										
										
											2009-05-02 12:36:44 +00:00
										 |  |  |         return NULL; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     /* if the next char is '.' followed by a digit then read all float digits */ | 
					
						
							| 
									
										
										
										
											2011-09-28 07:41:54 +02:00
										 |  |  |     if (idx < end_idx && PyUnicode_READ(kind, str, idx) == '.' && PyUnicode_READ(kind, str, idx + 1) >= '0' && PyUnicode_READ(kind, str, idx + 1) <= '9') { | 
					
						
							| 
									
										
										
										
											2009-05-02 12:36:44 +00:00
										 |  |  |         is_float = 1; | 
					
						
							|  |  |  |         idx += 2; | 
					
						
							| 
									
										
										
										
											2011-09-28 07:41:54 +02:00
										 |  |  |         while (idx <= end_idx && PyUnicode_READ(kind, str, idx) >= '0' && PyUnicode_READ(kind, str, idx) <= '9') idx++; | 
					
						
							| 
									
										
										
										
											2009-05-02 12:36:44 +00:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     /* if the next char is 'e' or 'E' then maybe read the exponent (or backtrack) */ | 
					
						
							| 
									
										
										
										
											2011-09-28 07:41:54 +02:00
										 |  |  |     if (idx < end_idx && (PyUnicode_READ(kind, str, idx) == 'e' || PyUnicode_READ(kind, str, idx) == 'E')) { | 
					
						
							| 
									
										
										
										
											2009-05-02 12:36:44 +00:00
										 |  |  |         Py_ssize_t e_start = idx; | 
					
						
							|  |  |  |         idx++; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         /* read an exponent sign if present */ | 
					
						
							| 
									
										
										
										
											2011-09-28 07:41:54 +02:00
										 |  |  |         if (idx < end_idx && (PyUnicode_READ(kind, str, idx) == '-' || PyUnicode_READ(kind, str, idx) == '+')) idx++; | 
					
						
							| 
									
										
										
										
											2009-05-02 12:36:44 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |         /* read all digits */ | 
					
						
							| 
									
										
										
										
											2011-09-28 07:41:54 +02:00
										 |  |  |         while (idx <= end_idx && PyUnicode_READ(kind, str, idx) >= '0' && PyUnicode_READ(kind, str, idx) <= '9') idx++; | 
					
						
							| 
									
										
										
										
											2009-05-02 12:36:44 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |         /* if we got a digit, then parse as float. if not, backtrack */ | 
					
						
							| 
									
										
										
										
											2011-09-28 07:41:54 +02:00
										 |  |  |         if (PyUnicode_READ(kind, str, idx - 1) >= '0' && PyUnicode_READ(kind, str, idx - 1) <= '9') { | 
					
						
							| 
									
										
										
										
											2009-05-02 12:36:44 +00:00
										 |  |  |             is_float = 1; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         else { | 
					
						
							|  |  |  |             idx = e_start; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-04-25 19:16:06 +02:00
										 |  |  |     if (is_float && s->parse_float != (PyObject *)&PyFloat_Type) | 
					
						
							|  |  |  |         custom_func = s->parse_float; | 
					
						
							|  |  |  |     else if (!is_float && s->parse_int != (PyObject *) &PyLong_Type) | 
					
						
							|  |  |  |         custom_func = s->parse_int; | 
					
						
							|  |  |  |     else | 
					
						
							|  |  |  |         custom_func = NULL; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (custom_func) { | 
					
						
							|  |  |  |         /* copy the section we determined to be a number */ | 
					
						
							| 
									
										
										
										
											2011-09-28 07:41:54 +02:00
										 |  |  |         numstr = PyUnicode_FromKindAndData(kind, | 
					
						
							| 
									
										
										
										
											2011-10-07 20:55:35 +02:00
										 |  |  |                                            (char*)str + kind * start, | 
					
						
							| 
									
										
										
										
											2011-09-28 07:41:54 +02:00
										 |  |  |                                            idx - start); | 
					
						
							| 
									
										
										
										
											2011-04-25 19:16:06 +02:00
										 |  |  |         if (numstr == NULL) | 
					
						
							|  |  |  |             return NULL; | 
					
						
							| 
									
										
										
										
											2020-02-11 17:46:57 +01:00
										 |  |  |         rval = PyObject_CallOneArg(custom_func, numstr); | 
					
						
							| 
									
										
										
										
											2009-05-02 12:36:44 +00:00
										 |  |  |     } | 
					
						
							|  |  |  |     else { | 
					
						
							| 
									
										
										
										
											2011-04-25 19:16:06 +02:00
										 |  |  |         Py_ssize_t i, n; | 
					
						
							|  |  |  |         char *buf; | 
					
						
							|  |  |  |         /* Straight conversion to ASCII, to avoid costly conversion of
 | 
					
						
							|  |  |  |            decimal unicode digits (which cannot appear here) */ | 
					
						
							|  |  |  |         n = idx - start; | 
					
						
							|  |  |  |         numstr = PyBytes_FromStringAndSize(NULL, n); | 
					
						
							|  |  |  |         if (numstr == NULL) | 
					
						
							|  |  |  |             return NULL; | 
					
						
							|  |  |  |         buf = PyBytes_AS_STRING(numstr); | 
					
						
							|  |  |  |         for (i = 0; i < n; i++) { | 
					
						
							| 
									
										
										
										
											2011-09-28 07:41:54 +02:00
										 |  |  |             buf[i] = (char) PyUnicode_READ(kind, str, i + start); | 
					
						
							| 
									
										
										
										
											2011-04-25 19:16:06 +02:00
										 |  |  |         } | 
					
						
							|  |  |  |         if (is_float) | 
					
						
							|  |  |  |             rval = PyFloat_FromString(numstr); | 
					
						
							|  |  |  |         else | 
					
						
							|  |  |  |             rval = PyLong_FromString(buf, NULL, 10); | 
					
						
							| 
									
										
										
										
											2009-05-02 12:36:44 +00:00
										 |  |  |     } | 
					
						
							|  |  |  |     Py_DECREF(numstr); | 
					
						
							|  |  |  |     *next_idx_ptr = idx; | 
					
						
							|  |  |  |     return rval; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static PyObject * | 
					
						
							|  |  |  | scan_once_unicode(PyScannerObject *s, PyObject *pystr, Py_ssize_t idx, Py_ssize_t *next_idx_ptr) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     /* Read one JSON term (of any kind) from PyUnicode pystr.
 | 
					
						
							|  |  |  |     idx is the index of the first character of the term | 
					
						
							|  |  |  |     *next_idx_ptr is a return-by-reference index to the first character after | 
					
						
							|  |  |  |         the number. | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     Returns a new PyObject representation of the term. | 
					
						
							|  |  |  |     */ | 
					
						
							| 
									
										
										
										
											2011-05-07 17:58:09 +03:00
										 |  |  |     PyObject *res; | 
					
						
							| 
									
										
										
										
											2020-04-11 10:48:40 +03:00
										 |  |  |     const void *str; | 
					
						
							| 
									
										
										
										
											2011-09-28 07:41:54 +02:00
										 |  |  |     int kind; | 
					
						
							|  |  |  |     Py_ssize_t length; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (PyUnicode_READY(pystr) == -1) | 
					
						
							|  |  |  |         return NULL; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     str = PyUnicode_DATA(pystr); | 
					
						
							|  |  |  |     kind = PyUnicode_KIND(pystr); | 
					
						
							|  |  |  |     length = PyUnicode_GET_LENGTH(pystr); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-04-14 11:45:21 -04:00
										 |  |  |     if (idx < 0) { | 
					
						
							| 
									
										
										
										
											2014-04-14 11:46:51 -04:00
										 |  |  |         PyErr_SetString(PyExc_ValueError, "idx cannot be negative"); | 
					
						
							| 
									
										
										
										
											2014-04-14 11:45:21 -04:00
										 |  |  |         return NULL; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     if (idx >= length) { | 
					
						
							| 
									
										
										
										
											2013-01-03 08:44:15 +02:00
										 |  |  |         raise_stop_iteration(idx); | 
					
						
							| 
									
										
										
										
											2009-05-02 12:36:44 +00:00
										 |  |  |         return NULL; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2011-09-28 07:41:54 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  |     switch (PyUnicode_READ(kind, str, idx)) { | 
					
						
							| 
									
										
										
										
											2009-05-02 12:36:44 +00:00
										 |  |  |         case '"': | 
					
						
							|  |  |  |             /* string */ | 
					
						
							| 
									
										
										
										
											2017-05-28 15:31:49 +03:00
										 |  |  |             return scanstring_unicode(pystr, idx + 1, s->strict, next_idx_ptr); | 
					
						
							| 
									
										
										
										
											2009-05-02 12:36:44 +00:00
										 |  |  |         case '{': | 
					
						
							|  |  |  |             /* object */ | 
					
						
							| 
									
										
										
										
											2022-05-04 13:30:23 +02:00
										 |  |  |             if (_Py_EnterRecursiveCall(" while decoding a JSON object " | 
					
						
							|  |  |  |                                        "from a unicode string")) | 
					
						
							| 
									
										
										
										
											2011-05-07 17:58:09 +03:00
										 |  |  |                 return NULL; | 
					
						
							|  |  |  |             res = _parse_object_unicode(s, pystr, idx + 1, next_idx_ptr); | 
					
						
							| 
									
										
										
										
											2022-05-04 13:30:23 +02:00
										 |  |  |             _Py_LeaveRecursiveCall(); | 
					
						
							| 
									
										
										
										
											2011-05-07 17:58:09 +03:00
										 |  |  |             return res; | 
					
						
							| 
									
										
										
										
											2009-05-02 12:36:44 +00:00
										 |  |  |         case '[': | 
					
						
							|  |  |  |             /* array */ | 
					
						
							| 
									
										
										
										
											2022-05-04 13:30:23 +02:00
										 |  |  |             if (_Py_EnterRecursiveCall(" while decoding a JSON array " | 
					
						
							|  |  |  |                                        "from a unicode string")) | 
					
						
							| 
									
										
										
										
											2011-05-07 17:58:09 +03:00
										 |  |  |                 return NULL; | 
					
						
							|  |  |  |             res = _parse_array_unicode(s, pystr, idx + 1, next_idx_ptr); | 
					
						
							| 
									
										
										
										
											2022-05-04 13:30:23 +02:00
										 |  |  |             _Py_LeaveRecursiveCall(); | 
					
						
							| 
									
										
										
										
											2011-05-07 17:58:09 +03:00
										 |  |  |             return res; | 
					
						
							| 
									
										
										
										
											2009-05-02 12:36:44 +00:00
										 |  |  |         case 'n': | 
					
						
							|  |  |  |             /* null */ | 
					
						
							| 
									
										
										
										
											2011-09-28 07:41:54 +02:00
										 |  |  |             if ((idx + 3 < length) && PyUnicode_READ(kind, str, idx + 1) == 'u' && PyUnicode_READ(kind, str, idx + 2) == 'l' && PyUnicode_READ(kind, str, idx + 3) == 'l') { | 
					
						
							| 
									
										
										
										
											2009-05-02 12:36:44 +00:00
										 |  |  |                 *next_idx_ptr = idx + 4; | 
					
						
							| 
									
										
										
										
											2017-01-23 10:23:58 +02:00
										 |  |  |                 Py_RETURN_NONE; | 
					
						
							| 
									
										
										
										
											2009-05-02 12:36:44 +00:00
										 |  |  |             } | 
					
						
							|  |  |  |             break; | 
					
						
							|  |  |  |         case 't': | 
					
						
							|  |  |  |             /* true */ | 
					
						
							| 
									
										
										
										
											2011-09-28 07:41:54 +02:00
										 |  |  |             if ((idx + 3 < length) && PyUnicode_READ(kind, str, idx + 1) == 'r' && PyUnicode_READ(kind, str, idx + 2) == 'u' && PyUnicode_READ(kind, str, idx + 3) == 'e') { | 
					
						
							| 
									
										
										
										
											2009-05-02 12:36:44 +00:00
										 |  |  |                 *next_idx_ptr = idx + 4; | 
					
						
							| 
									
										
										
										
											2017-01-23 10:23:58 +02:00
										 |  |  |                 Py_RETURN_TRUE; | 
					
						
							| 
									
										
										
										
											2009-05-02 12:36:44 +00:00
										 |  |  |             } | 
					
						
							|  |  |  |             break; | 
					
						
							|  |  |  |         case 'f': | 
					
						
							|  |  |  |             /* false */ | 
					
						
							| 
									
										
										
										
											2011-10-11 21:56:19 +02:00
										 |  |  |             if ((idx + 4 < length) && PyUnicode_READ(kind, str, idx + 1) == 'a' && | 
					
						
							|  |  |  |                 PyUnicode_READ(kind, str, idx + 2) == 'l' && | 
					
						
							|  |  |  |                 PyUnicode_READ(kind, str, idx + 3) == 's' && | 
					
						
							| 
									
										
										
										
											2011-09-28 07:41:54 +02:00
										 |  |  |                 PyUnicode_READ(kind, str, idx + 4) == 'e') { | 
					
						
							| 
									
										
										
										
											2009-05-02 12:36:44 +00:00
										 |  |  |                 *next_idx_ptr = idx + 5; | 
					
						
							| 
									
										
										
										
											2017-01-23 10:23:58 +02:00
										 |  |  |                 Py_RETURN_FALSE; | 
					
						
							| 
									
										
										
										
											2009-05-02 12:36:44 +00:00
										 |  |  |             } | 
					
						
							|  |  |  |             break; | 
					
						
							|  |  |  |         case 'N': | 
					
						
							|  |  |  |             /* NaN */ | 
					
						
							| 
									
										
										
										
											2011-10-11 21:56:19 +02:00
										 |  |  |             if ((idx + 2 < length) && PyUnicode_READ(kind, str, idx + 1) == 'a' && | 
					
						
							| 
									
										
										
										
											2011-09-28 07:41:54 +02:00
										 |  |  |                 PyUnicode_READ(kind, str, idx + 2) == 'N') { | 
					
						
							| 
									
										
										
										
											2009-05-02 12:36:44 +00:00
										 |  |  |                 return _parse_constant(s, "NaN", idx, next_idx_ptr); | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |             break; | 
					
						
							|  |  |  |         case 'I': | 
					
						
							|  |  |  |             /* Infinity */ | 
					
						
							| 
									
										
										
										
											2011-10-11 21:56:19 +02:00
										 |  |  |             if ((idx + 7 < length) && PyUnicode_READ(kind, str, idx + 1) == 'n' && | 
					
						
							|  |  |  |                 PyUnicode_READ(kind, str, idx + 2) == 'f' && | 
					
						
							|  |  |  |                 PyUnicode_READ(kind, str, idx + 3) == 'i' && | 
					
						
							| 
									
										
										
										
											2011-09-28 07:41:54 +02:00
										 |  |  |                 PyUnicode_READ(kind, str, idx + 4) == 'n' && | 
					
						
							| 
									
										
										
										
											2011-10-11 21:56:19 +02:00
										 |  |  |                 PyUnicode_READ(kind, str, idx + 5) == 'i' && | 
					
						
							|  |  |  |                 PyUnicode_READ(kind, str, idx + 6) == 't' && | 
					
						
							| 
									
										
										
										
											2011-09-28 07:41:54 +02:00
										 |  |  |                 PyUnicode_READ(kind, str, idx + 7) == 'y') { | 
					
						
							| 
									
										
										
										
											2009-05-02 12:36:44 +00:00
										 |  |  |                 return _parse_constant(s, "Infinity", idx, next_idx_ptr); | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |             break; | 
					
						
							|  |  |  |         case '-': | 
					
						
							|  |  |  |             /* -Infinity */ | 
					
						
							| 
									
										
										
										
											2011-10-11 21:56:19 +02:00
										 |  |  |             if ((idx + 8 < length) && PyUnicode_READ(kind, str, idx + 1) == 'I' && | 
					
						
							| 
									
										
										
										
											2011-09-28 07:41:54 +02:00
										 |  |  |                 PyUnicode_READ(kind, str, idx + 2) == 'n' && | 
					
						
							|  |  |  |                 PyUnicode_READ(kind, str, idx + 3) == 'f' && | 
					
						
							| 
									
										
										
										
											2011-10-11 21:56:19 +02:00
										 |  |  |                 PyUnicode_READ(kind, str, idx + 4) == 'i' && | 
					
						
							| 
									
										
										
										
											2011-09-28 07:41:54 +02:00
										 |  |  |                 PyUnicode_READ(kind, str, idx + 5) == 'n' && | 
					
						
							| 
									
										
										
										
											2011-10-11 21:56:19 +02:00
										 |  |  |                 PyUnicode_READ(kind, str, idx + 6) == 'i' && | 
					
						
							|  |  |  |                 PyUnicode_READ(kind, str, idx + 7) == 't' && | 
					
						
							| 
									
										
										
										
											2011-09-28 07:41:54 +02:00
										 |  |  |                 PyUnicode_READ(kind, str, idx + 8) == 'y') { | 
					
						
							| 
									
										
										
										
											2009-05-02 12:36:44 +00:00
										 |  |  |                 return _parse_constant(s, "-Infinity", idx, next_idx_ptr); | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |             break; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     /* Didn't find a string, object, array, or named constant. Look for a number. */ | 
					
						
							|  |  |  |     return _match_number_unicode(s, pystr, idx, next_idx_ptr); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static PyObject * | 
					
						
							| 
									
										
										
										
											2020-03-27 19:59:59 +09:00
										 |  |  | scanner_call(PyScannerObject *self, PyObject *args, PyObject *kwds) | 
					
						
							| 
									
										
										
										
											2009-05-02 12:36:44 +00:00
										 |  |  | { | 
					
						
							|  |  |  |     /* Python callable interface to scan_once_{str,unicode} */ | 
					
						
							|  |  |  |     PyObject *pystr; | 
					
						
							|  |  |  |     PyObject *rval; | 
					
						
							|  |  |  |     Py_ssize_t idx; | 
					
						
							|  |  |  |     Py_ssize_t next_idx = -1; | 
					
						
							|  |  |  |     static char *kwlist[] = {"string", "idx", NULL}; | 
					
						
							| 
									
										
										
										
											2012-12-01 19:34:16 +01:00
										 |  |  |     if (!PyArg_ParseTupleAndKeywords(args, kwds, "On:scan_once", kwlist, &pystr, &idx)) | 
					
						
							| 
									
										
										
										
											2009-05-02 12:36:44 +00:00
										 |  |  |         return NULL; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (PyUnicode_Check(pystr)) { | 
					
						
							| 
									
										
										
										
											2020-03-27 19:59:59 +09:00
										 |  |  |         rval = scan_once_unicode(self, pystr, idx, &next_idx); | 
					
						
							| 
									
										
										
										
											2009-05-02 12:36:44 +00:00
										 |  |  |     } | 
					
						
							|  |  |  |     else { | 
					
						
							|  |  |  |         PyErr_Format(PyExc_TypeError, | 
					
						
							|  |  |  |                  "first argument must be a string, not %.80s", | 
					
						
							|  |  |  |                  Py_TYPE(pystr)->tp_name); | 
					
						
							|  |  |  |         return NULL; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2020-03-27 19:59:59 +09:00
										 |  |  |     PyDict_Clear(self->memo); | 
					
						
							| 
									
										
										
										
											2010-09-04 20:16:53 +00:00
										 |  |  |     if (rval == NULL) | 
					
						
							|  |  |  |         return NULL; | 
					
						
							| 
									
										
										
										
											2009-05-02 12:36:44 +00:00
										 |  |  |     return _build_rval_index_tuple(rval, next_idx); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static PyObject * | 
					
						
							|  |  |  | scanner_new(PyTypeObject *type, PyObject *args, PyObject *kwds) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     PyScannerObject *s; | 
					
						
							|  |  |  |     PyObject *ctx; | 
					
						
							| 
									
										
										
										
											2017-05-28 15:31:49 +03:00
										 |  |  |     PyObject *strict; | 
					
						
							| 
									
										
										
										
											2009-05-02 12:36:44 +00:00
										 |  |  |     static char *kwlist[] = {"context", NULL}; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (!PyArg_ParseTupleAndKeywords(args, kwds, "O:make_scanner", kwlist, &ctx)) | 
					
						
							| 
									
										
										
										
											2017-05-05 10:08:49 +03:00
										 |  |  |         return NULL; | 
					
						
							| 
									
										
										
										
											2009-05-02 12:36:44 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-05-05 10:08:49 +03:00
										 |  |  |     s = (PyScannerObject *)type->tp_alloc(type, 0); | 
					
						
							|  |  |  |     if (s == NULL) { | 
					
						
							|  |  |  |         return NULL; | 
					
						
							| 
									
										
										
										
											2010-09-04 20:16:53 +00:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-05-05 10:08:49 +03:00
										 |  |  |     s->memo = PyDict_New(); | 
					
						
							|  |  |  |     if (s->memo == NULL) | 
					
						
							|  |  |  |         goto bail; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-05-02 12:36:44 +00:00
										 |  |  |     /* All of these will fail "gracefully" so we don't need to verify them */ | 
					
						
							| 
									
										
										
										
											2017-05-28 15:31:49 +03:00
										 |  |  |     strict = PyObject_GetAttrString(ctx, "strict"); | 
					
						
							|  |  |  |     if (strict == NULL) | 
					
						
							|  |  |  |         goto bail; | 
					
						
							|  |  |  |     s->strict = PyObject_IsTrue(strict); | 
					
						
							|  |  |  |     Py_DECREF(strict); | 
					
						
							|  |  |  |     if (s->strict < 0) | 
					
						
							| 
									
										
										
										
											2009-05-02 12:36:44 +00:00
										 |  |  |         goto bail; | 
					
						
							|  |  |  |     s->object_hook = PyObject_GetAttrString(ctx, "object_hook"); | 
					
						
							|  |  |  |     if (s->object_hook == NULL) | 
					
						
							|  |  |  |         goto bail; | 
					
						
							|  |  |  |     s->object_pairs_hook = PyObject_GetAttrString(ctx, "object_pairs_hook"); | 
					
						
							|  |  |  |     if (s->object_pairs_hook == NULL) | 
					
						
							|  |  |  |         goto bail; | 
					
						
							|  |  |  |     s->parse_float = PyObject_GetAttrString(ctx, "parse_float"); | 
					
						
							|  |  |  |     if (s->parse_float == NULL) | 
					
						
							|  |  |  |         goto bail; | 
					
						
							|  |  |  |     s->parse_int = PyObject_GetAttrString(ctx, "parse_int"); | 
					
						
							|  |  |  |     if (s->parse_int == NULL) | 
					
						
							|  |  |  |         goto bail; | 
					
						
							|  |  |  |     s->parse_constant = PyObject_GetAttrString(ctx, "parse_constant"); | 
					
						
							|  |  |  |     if (s->parse_constant == NULL) | 
					
						
							|  |  |  |         goto bail; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-05-05 10:08:49 +03:00
										 |  |  |     return (PyObject *)s; | 
					
						
							| 
									
										
										
										
											2009-05-02 12:36:44 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | bail: | 
					
						
							| 
									
										
										
										
											2017-05-05 10:08:49 +03:00
										 |  |  |     Py_DECREF(s); | 
					
						
							|  |  |  |     return NULL; | 
					
						
							| 
									
										
										
										
											2009-05-02 12:36:44 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | PyDoc_STRVAR(scanner_doc, "JSON scanner object"); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-03-27 19:59:59 +09:00
										 |  |  | static PyType_Slot PyScannerType_slots[] = { | 
					
						
							|  |  |  |     {Py_tp_doc, (void *)scanner_doc}, | 
					
						
							|  |  |  |     {Py_tp_dealloc, scanner_dealloc}, | 
					
						
							|  |  |  |     {Py_tp_call, scanner_call}, | 
					
						
							|  |  |  |     {Py_tp_traverse, scanner_traverse}, | 
					
						
							|  |  |  |     {Py_tp_clear, scanner_clear}, | 
					
						
							|  |  |  |     {Py_tp_members, scanner_members}, | 
					
						
							|  |  |  |     {Py_tp_new, scanner_new}, | 
					
						
							|  |  |  |     {0, 0} | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static PyType_Spec PyScannerType_spec = { | 
					
						
							|  |  |  |     .name = "_json.Scanner", | 
					
						
							|  |  |  |     .basicsize = sizeof(PyScannerObject), | 
					
						
							|  |  |  |     .itemsize = 0, | 
					
						
							|  |  |  |     .flags = Py_TPFLAGS_DEFAULT | Py_TPFLAGS_HAVE_GC, | 
					
						
							|  |  |  |     .slots = PyScannerType_slots, | 
					
						
							| 
									
										
										
										
											2009-05-02 12:36:44 +00:00
										 |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static PyObject * | 
					
						
							|  |  |  | encoder_new(PyTypeObject *type, PyObject *args, PyObject *kwds) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     static char *kwlist[] = {"markers", "default", "encoder", "indent", "key_separator", "item_separator", "sort_keys", "skipkeys", "allow_nan", NULL}; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     PyEncoderObject *s; | 
					
						
							| 
									
										
										
										
											2009-12-08 15:57:31 +00:00
										 |  |  |     PyObject *markers, *defaultfn, *encoder, *indent, *key_separator; | 
					
						
							| 
									
										
										
										
											2017-05-28 15:31:49 +03:00
										 |  |  |     PyObject *item_separator; | 
					
						
							|  |  |  |     int sort_keys, skipkeys, allow_nan; | 
					
						
							| 
									
										
										
										
											2009-05-02 12:36:44 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-05-28 15:31:49 +03:00
										 |  |  |     if (!PyArg_ParseTupleAndKeywords(args, kwds, "OOOOUUppp:make_encoder", kwlist, | 
					
						
							| 
									
										
										
										
											2015-07-26 09:01:22 +03:00
										 |  |  |         &markers, &defaultfn, &encoder, &indent, | 
					
						
							|  |  |  |         &key_separator, &item_separator, | 
					
						
							| 
									
										
										
										
											2009-12-08 15:57:31 +00:00
										 |  |  |         &sort_keys, &skipkeys, &allow_nan)) | 
					
						
							| 
									
										
										
										
											2017-05-05 10:08:49 +03:00
										 |  |  |         return NULL; | 
					
						
							| 
									
										
										
										
											2009-05-02 12:36:44 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-07-26 09:01:22 +03:00
										 |  |  |     if (markers != Py_None && !PyDict_Check(markers)) { | 
					
						
							|  |  |  |         PyErr_Format(PyExc_TypeError, | 
					
						
							|  |  |  |                      "make_encoder() argument 1 must be dict or None, " | 
					
						
							|  |  |  |                      "not %.200s", Py_TYPE(markers)->tp_name); | 
					
						
							| 
									
										
										
										
											2017-05-05 10:08:49 +03:00
										 |  |  |         return NULL; | 
					
						
							| 
									
										
										
										
											2015-07-26 09:01:22 +03:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-05-05 10:08:49 +03:00
										 |  |  |     s = (PyEncoderObject *)type->tp_alloc(type, 0); | 
					
						
							|  |  |  |     if (s == NULL) | 
					
						
							|  |  |  |         return NULL; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-12-08 15:57:31 +00:00
										 |  |  |     s->markers = markers; | 
					
						
							|  |  |  |     s->defaultfn = defaultfn; | 
					
						
							|  |  |  |     s->encoder = encoder; | 
					
						
							|  |  |  |     s->indent = indent; | 
					
						
							|  |  |  |     s->key_separator = key_separator; | 
					
						
							|  |  |  |     s->item_separator = item_separator; | 
					
						
							|  |  |  |     s->sort_keys = sort_keys; | 
					
						
							|  |  |  |     s->skipkeys = skipkeys; | 
					
						
							| 
									
										
										
										
											2017-05-28 15:31:49 +03:00
										 |  |  |     s->allow_nan = allow_nan; | 
					
						
							| 
									
										
										
										
											2015-01-11 16:41:01 +01:00
										 |  |  |     s->fast_encode = NULL; | 
					
						
							|  |  |  |     if (PyCFunction_Check(s->encoder)) { | 
					
						
							|  |  |  |         PyCFunction f = PyCFunction_GetFunction(s->encoder); | 
					
						
							|  |  |  |         if (f == (PyCFunction)py_encode_basestring_ascii || | 
					
						
							|  |  |  |                 f == (PyCFunction)py_encode_basestring) { | 
					
						
							|  |  |  |             s->fast_encode = f; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2009-12-08 15:57:31 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-05-02 12:36:44 +00:00
										 |  |  |     Py_INCREF(s->markers); | 
					
						
							|  |  |  |     Py_INCREF(s->defaultfn); | 
					
						
							|  |  |  |     Py_INCREF(s->encoder); | 
					
						
							|  |  |  |     Py_INCREF(s->indent); | 
					
						
							|  |  |  |     Py_INCREF(s->key_separator); | 
					
						
							|  |  |  |     Py_INCREF(s->item_separator); | 
					
						
							| 
									
										
										
										
											2017-05-05 10:08:49 +03:00
										 |  |  |     return (PyObject *)s; | 
					
						
							| 
									
										
										
										
											2009-05-02 12:36:44 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static PyObject * | 
					
						
							| 
									
										
										
										
											2020-03-27 19:59:59 +09:00
										 |  |  | encoder_call(PyEncoderObject *self, PyObject *args, PyObject *kwds) | 
					
						
							| 
									
										
										
										
											2009-05-02 12:36:44 +00:00
										 |  |  | { | 
					
						
							|  |  |  |     /* Python callable interface to encode_listencode_obj */ | 
					
						
							|  |  |  |     static char *kwlist[] = {"obj", "_current_indent_level", NULL}; | 
					
						
							| 
									
										
										
										
											2022-07-27 11:43:34 +03:00
										 |  |  |     PyObject *obj, *result; | 
					
						
							| 
									
										
										
										
											2009-05-02 12:36:44 +00:00
										 |  |  |     Py_ssize_t indent_level; | 
					
						
							| 
									
										
										
										
											2022-07-27 11:43:34 +03:00
										 |  |  |     _PyUnicodeWriter writer; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-12-01 19:34:16 +01:00
										 |  |  |     if (!PyArg_ParseTupleAndKeywords(args, kwds, "On:_iterencode", kwlist, | 
					
						
							|  |  |  |         &obj, &indent_level)) | 
					
						
							| 
									
										
										
										
											2009-05-02 12:36:44 +00:00
										 |  |  |         return NULL; | 
					
						
							| 
									
										
										
										
											2022-07-27 11:43:34 +03:00
										 |  |  | 
 | 
					
						
							|  |  |  |     _PyUnicodeWriter_Init(&writer); | 
					
						
							|  |  |  |     writer.overallocate = 1; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (encoder_listencode_obj(self, &writer, obj, indent_level)) { | 
					
						
							|  |  |  |         _PyUnicodeWriter_Dealloc(&writer); | 
					
						
							| 
									
										
										
										
											2009-05-02 12:36:44 +00:00
										 |  |  |         return NULL; | 
					
						
							| 
									
										
										
										
											2022-07-27 11:43:34 +03:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     result = PyTuple_New(1); | 
					
						
							|  |  |  |     if (result == NULL || | 
					
						
							|  |  |  |             PyTuple_SetItem(result, 0, _PyUnicodeWriter_Finish(&writer)) < 0) { | 
					
						
							|  |  |  |         Py_XDECREF(result); | 
					
						
							| 
									
										
										
										
											2009-05-02 12:36:44 +00:00
										 |  |  |         return NULL; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2022-07-27 11:43:34 +03:00
										 |  |  |     return result; | 
					
						
							| 
									
										
										
										
											2009-05-02 12:36:44 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static PyObject * | 
					
						
							|  |  |  | _encoded_const(PyObject *obj) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     /* Return the JSON string representation of None, True, False */ | 
					
						
							|  |  |  |     if (obj == Py_None) { | 
					
						
							| 
									
										
										
										
											2022-11-02 21:33:38 +05:30
										 |  |  |         return Py_NewRef(&_Py_ID(null)); | 
					
						
							| 
									
										
										
										
											2009-05-02 12:36:44 +00:00
										 |  |  |     } | 
					
						
							|  |  |  |     else if (obj == Py_True) { | 
					
						
							| 
									
										
										
										
											2022-11-02 21:33:38 +05:30
										 |  |  |         return Py_NewRef(&_Py_ID(true)); | 
					
						
							| 
									
										
										
										
											2009-05-02 12:36:44 +00:00
										 |  |  |     } | 
					
						
							|  |  |  |     else if (obj == Py_False) { | 
					
						
							| 
									
										
										
										
											2022-11-02 21:33:38 +05:30
										 |  |  |         return Py_NewRef(&_Py_ID(false)); | 
					
						
							| 
									
										
										
										
											2009-05-02 12:36:44 +00:00
										 |  |  |     } | 
					
						
							|  |  |  |     else { | 
					
						
							|  |  |  |         PyErr_SetString(PyExc_ValueError, "not a const"); | 
					
						
							|  |  |  |         return NULL; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static PyObject * | 
					
						
							|  |  |  | encoder_encode_float(PyEncoderObject *s, PyObject *obj) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2016-04-10 14:41:19 +03:00
										 |  |  |     /* Return the JSON representation of a PyFloat. */ | 
					
						
							| 
									
										
										
										
											2009-05-02 12:36:44 +00:00
										 |  |  |     double i = PyFloat_AS_DOUBLE(obj); | 
					
						
							|  |  |  |     if (!Py_IS_FINITE(i)) { | 
					
						
							|  |  |  |         if (!s->allow_nan) { | 
					
						
							| 
									
										
										
										
											2013-08-10 13:01:45 -07:00
										 |  |  |             PyErr_SetString( | 
					
						
							|  |  |  |                     PyExc_ValueError, | 
					
						
							|  |  |  |                     "Out of range float values are not JSON compliant" | 
					
						
							|  |  |  |                     ); | 
					
						
							| 
									
										
										
										
											2009-05-02 12:36:44 +00:00
										 |  |  |             return NULL; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         if (i > 0) { | 
					
						
							|  |  |  |             return PyUnicode_FromString("Infinity"); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         else if (i < 0) { | 
					
						
							|  |  |  |             return PyUnicode_FromString("-Infinity"); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         else { | 
					
						
							|  |  |  |             return PyUnicode_FromString("NaN"); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2016-04-10 14:41:19 +03:00
										 |  |  |     return PyFloat_Type.tp_repr(obj); | 
					
						
							| 
									
										
										
										
											2009-05-02 12:36:44 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static PyObject * | 
					
						
							|  |  |  | encoder_encode_string(PyEncoderObject *s, PyObject *obj) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     /* Return the JSON representation of a string */ | 
					
						
							| 
									
										
										
										
											2017-09-24 12:07:12 +03:00
										 |  |  |     PyObject *encoded; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (s->fast_encode) { | 
					
						
							| 
									
										
										
										
											2015-01-11 16:41:01 +01:00
										 |  |  |         return s->fast_encode(NULL, obj); | 
					
						
							| 
									
										
										
										
											2017-09-24 12:07:12 +03:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2020-02-11 17:46:57 +01:00
										 |  |  |     encoded = PyObject_CallOneArg(s->encoder, obj); | 
					
						
							| 
									
										
										
										
											2017-09-24 12:07:12 +03:00
										 |  |  |     if (encoded != NULL && !PyUnicode_Check(encoded)) { | 
					
						
							|  |  |  |         PyErr_Format(PyExc_TypeError, | 
					
						
							|  |  |  |                      "encoder() must return a string, not %.80s", | 
					
						
							|  |  |  |                      Py_TYPE(encoded)->tp_name); | 
					
						
							|  |  |  |         Py_DECREF(encoded); | 
					
						
							|  |  |  |         return NULL; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     return encoded; | 
					
						
							| 
									
										
										
										
											2009-05-02 12:36:44 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static int | 
					
						
							| 
									
										
										
										
											2022-07-27 11:43:34 +03:00
										 |  |  | _steal_accumulate(_PyUnicodeWriter *writer, PyObject *stolen) | 
					
						
							| 
									
										
										
										
											2009-05-02 12:36:44 +00:00
										 |  |  | { | 
					
						
							|  |  |  |     /* Append stolen and then decrement its reference count */ | 
					
						
							| 
									
										
										
										
											2022-07-27 11:43:34 +03:00
										 |  |  |     int rval = _PyUnicodeWriter_WriteStr(writer, stolen); | 
					
						
							| 
									
										
										
										
											2009-05-02 12:36:44 +00:00
										 |  |  |     Py_DECREF(stolen); | 
					
						
							|  |  |  |     return rval; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static int | 
					
						
							| 
									
										
										
										
											2022-07-27 11:43:34 +03:00
										 |  |  | encoder_listencode_obj(PyEncoderObject *s, _PyUnicodeWriter *writer, | 
					
						
							| 
									
										
										
										
											2011-08-19 18:03:14 +02:00
										 |  |  |                        PyObject *obj, Py_ssize_t indent_level) | 
					
						
							| 
									
										
										
										
											2009-05-02 12:36:44 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2011-08-19 18:03:14 +02:00
										 |  |  |     /* Encode Python object obj to a JSON term */ | 
					
						
							| 
									
										
										
										
											2009-05-02 12:36:44 +00:00
										 |  |  |     PyObject *newobj; | 
					
						
							|  |  |  |     int rv; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-07-27 11:43:34 +03:00
										 |  |  |     if (obj == Py_None) { | 
					
						
							|  |  |  |       return _PyUnicodeWriter_WriteASCIIString(writer, "null", 4); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     else if (obj == Py_True) { | 
					
						
							|  |  |  |       return _PyUnicodeWriter_WriteASCIIString(writer, "true", 4); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     else if (obj == Py_False) { | 
					
						
							|  |  |  |       return _PyUnicodeWriter_WriteASCIIString(writer, "false", 5); | 
					
						
							| 
									
										
										
										
											2009-05-02 12:36:44 +00:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2022-07-27 11:43:34 +03:00
										 |  |  |     else if (PyUnicode_Check(obj)) { | 
					
						
							| 
									
										
										
										
											2009-05-02 12:36:44 +00:00
										 |  |  |         PyObject *encoded = encoder_encode_string(s, obj); | 
					
						
							|  |  |  |         if (encoded == NULL) | 
					
						
							|  |  |  |             return -1; | 
					
						
							| 
									
										
										
										
											2022-07-27 11:43:34 +03:00
										 |  |  |         return _steal_accumulate(writer, encoded); | 
					
						
							| 
									
										
										
										
											2009-05-02 12:36:44 +00:00
										 |  |  |     } | 
					
						
							|  |  |  |     else if (PyLong_Check(obj)) { | 
					
						
							| 
									
										
										
										
											2019-05-06 22:29:40 +03:00
										 |  |  |         PyObject *encoded = PyLong_Type.tp_repr(obj); | 
					
						
							| 
									
										
										
										
											2009-05-02 12:36:44 +00:00
										 |  |  |         if (encoded == NULL) | 
					
						
							|  |  |  |             return -1; | 
					
						
							| 
									
										
										
										
											2022-07-27 11:43:34 +03:00
										 |  |  |         return _steal_accumulate(writer, encoded); | 
					
						
							| 
									
										
										
										
											2009-05-02 12:36:44 +00:00
										 |  |  |     } | 
					
						
							|  |  |  |     else if (PyFloat_Check(obj)) { | 
					
						
							|  |  |  |         PyObject *encoded = encoder_encode_float(s, obj); | 
					
						
							|  |  |  |         if (encoded == NULL) | 
					
						
							|  |  |  |             return -1; | 
					
						
							| 
									
										
										
										
											2022-07-27 11:43:34 +03:00
										 |  |  |         return _steal_accumulate(writer, encoded); | 
					
						
							| 
									
										
										
										
											2009-05-02 12:36:44 +00:00
										 |  |  |     } | 
					
						
							|  |  |  |     else if (PyList_Check(obj) || PyTuple_Check(obj)) { | 
					
						
							| 
									
										
										
										
											2022-05-04 13:30:23 +02:00
										 |  |  |         if (_Py_EnterRecursiveCall(" while encoding a JSON object")) | 
					
						
							| 
									
										
										
										
											2011-05-11 01:02:56 +03:00
										 |  |  |             return -1; | 
					
						
							| 
									
										
										
										
											2022-07-27 11:43:34 +03:00
										 |  |  |         rv = encoder_listencode_list(s, writer, obj, indent_level); | 
					
						
							| 
									
										
										
										
											2022-05-04 13:30:23 +02:00
										 |  |  |         _Py_LeaveRecursiveCall(); | 
					
						
							| 
									
										
										
										
											2011-05-11 01:02:56 +03:00
										 |  |  |         return rv; | 
					
						
							| 
									
										
										
										
											2009-05-02 12:36:44 +00:00
										 |  |  |     } | 
					
						
							|  |  |  |     else if (PyDict_Check(obj)) { | 
					
						
							| 
									
										
										
										
											2022-05-04 13:30:23 +02:00
										 |  |  |         if (_Py_EnterRecursiveCall(" while encoding a JSON object")) | 
					
						
							| 
									
										
										
										
											2011-05-11 01:02:56 +03:00
										 |  |  |             return -1; | 
					
						
							| 
									
										
										
										
											2022-07-27 11:43:34 +03:00
										 |  |  |         rv = encoder_listencode_dict(s, writer, obj, indent_level); | 
					
						
							| 
									
										
										
										
											2022-05-04 13:30:23 +02:00
										 |  |  |         _Py_LeaveRecursiveCall(); | 
					
						
							| 
									
										
										
										
											2011-05-11 01:02:56 +03:00
										 |  |  |         return rv; | 
					
						
							| 
									
										
										
										
											2009-05-02 12:36:44 +00:00
										 |  |  |     } | 
					
						
							|  |  |  |     else { | 
					
						
							|  |  |  |         PyObject *ident = NULL; | 
					
						
							|  |  |  |         if (s->markers != Py_None) { | 
					
						
							|  |  |  |             int has_key; | 
					
						
							|  |  |  |             ident = PyLong_FromVoidPtr(obj); | 
					
						
							|  |  |  |             if (ident == NULL) | 
					
						
							|  |  |  |                 return -1; | 
					
						
							|  |  |  |             has_key = PyDict_Contains(s->markers, ident); | 
					
						
							|  |  |  |             if (has_key) { | 
					
						
							|  |  |  |                 if (has_key != -1) | 
					
						
							|  |  |  |                     PyErr_SetString(PyExc_ValueError, "Circular reference detected"); | 
					
						
							|  |  |  |                 Py_DECREF(ident); | 
					
						
							|  |  |  |                 return -1; | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |             if (PyDict_SetItem(s->markers, ident, obj)) { | 
					
						
							|  |  |  |                 Py_DECREF(ident); | 
					
						
							|  |  |  |                 return -1; | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											2020-02-11 17:46:57 +01:00
										 |  |  |         newobj = PyObject_CallOneArg(s->defaultfn, obj); | 
					
						
							| 
									
										
										
										
											2009-05-02 12:36:44 +00:00
										 |  |  |         if (newobj == NULL) { | 
					
						
							|  |  |  |             Py_XDECREF(ident); | 
					
						
							|  |  |  |             return -1; | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											2011-05-11 01:02:56 +03:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-05-04 13:30:23 +02:00
										 |  |  |         if (_Py_EnterRecursiveCall(" while encoding a JSON object")) { | 
					
						
							| 
									
										
										
										
											2017-01-03 11:17:44 +02:00
										 |  |  |             Py_DECREF(newobj); | 
					
						
							|  |  |  |             Py_XDECREF(ident); | 
					
						
							| 
									
										
										
										
											2011-05-11 01:02:56 +03:00
										 |  |  |             return -1; | 
					
						
							| 
									
										
										
										
											2017-01-03 11:17:44 +02:00
										 |  |  |         } | 
					
						
							| 
									
										
										
										
											2022-07-27 11:43:34 +03:00
										 |  |  |         rv = encoder_listencode_obj(s, writer, newobj, indent_level); | 
					
						
							| 
									
										
										
										
											2022-05-04 13:30:23 +02:00
										 |  |  |         _Py_LeaveRecursiveCall(); | 
					
						
							| 
									
										
										
										
											2011-05-11 01:02:56 +03:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-05-02 12:36:44 +00:00
										 |  |  |         Py_DECREF(newobj); | 
					
						
							|  |  |  |         if (rv) { | 
					
						
							|  |  |  |             Py_XDECREF(ident); | 
					
						
							|  |  |  |             return -1; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         if (ident != NULL) { | 
					
						
							|  |  |  |             if (PyDict_DelItem(s->markers, ident)) { | 
					
						
							|  |  |  |                 Py_XDECREF(ident); | 
					
						
							|  |  |  |                 return -1; | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |             Py_XDECREF(ident); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         return rv; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-08-06 09:55:24 +03:00
										 |  |  | static int | 
					
						
							|  |  |  | encoder_encode_key_value(PyEncoderObject *s, _PyUnicodeWriter *writer, bool *first, | 
					
						
							|  |  |  |                          PyObject *key, PyObject *value, Py_ssize_t indent_level) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     PyObject *keystr = NULL; | 
					
						
							|  |  |  |     PyObject *encoded; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (PyUnicode_Check(key)) { | 
					
						
							|  |  |  |         Py_INCREF(key); | 
					
						
							|  |  |  |         keystr = key; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     else if (PyFloat_Check(key)) { | 
					
						
							|  |  |  |         keystr = encoder_encode_float(s, key); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     else if (key == Py_True || key == Py_False || key == Py_None) { | 
					
						
							|  |  |  |                     /* This must come before the PyLong_Check because
 | 
					
						
							|  |  |  |                        True and False are also 1 and 0.*/ | 
					
						
							|  |  |  |         keystr = _encoded_const(key); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     else if (PyLong_Check(key)) { | 
					
						
							|  |  |  |         keystr = PyLong_Type.tp_repr(key); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     else if (s->skipkeys) { | 
					
						
							|  |  |  |         return 0; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     else { | 
					
						
							|  |  |  |         PyErr_Format(PyExc_TypeError, | 
					
						
							|  |  |  |                      "keys must be str, int, float, bool or None, " | 
					
						
							|  |  |  |                      "not %.100s", Py_TYPE(key)->tp_name); | 
					
						
							|  |  |  |         return -1; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (keystr == NULL) { | 
					
						
							|  |  |  |         return -1; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (*first) { | 
					
						
							|  |  |  |         *first = false; | 
					
						
							| 
									
										
										
										
											2022-11-02 21:33:38 +05:30
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2022-08-06 09:55:24 +03:00
										 |  |  |     else { | 
					
						
							|  |  |  |         if (_PyUnicodeWriter_WriteStr(writer, s->item_separator) < 0) { | 
					
						
							|  |  |  |             Py_DECREF(keystr); | 
					
						
							|  |  |  |             return -1; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     encoded = encoder_encode_string(s, keystr); | 
					
						
							|  |  |  |     Py_DECREF(keystr); | 
					
						
							|  |  |  |     if (encoded == NULL) { | 
					
						
							|  |  |  |         return -1; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (_steal_accumulate(writer, encoded) < 0) { | 
					
						
							|  |  |  |         return -1; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     if (_PyUnicodeWriter_WriteStr(writer, s->key_separator) < 0) { | 
					
						
							|  |  |  |         return -1; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     if (encoder_listencode_obj(s, writer, value, indent_level) < 0) { | 
					
						
							|  |  |  |         return -1; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     return 0; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-05-02 12:36:44 +00:00
										 |  |  | static int | 
					
						
							| 
									
										
										
										
											2022-07-27 11:43:34 +03:00
										 |  |  | encoder_listencode_dict(PyEncoderObject *s, _PyUnicodeWriter *writer, | 
					
						
							| 
									
										
										
										
											2011-08-19 18:03:14 +02:00
										 |  |  |                         PyObject *dct, Py_ssize_t indent_level) | 
					
						
							| 
									
										
										
										
											2009-05-02 12:36:44 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2011-08-19 18:03:14 +02:00
										 |  |  |     /* Encode Python dict dct a JSON term */ | 
					
						
							| 
									
										
										
										
											2009-05-02 12:36:44 +00:00
										 |  |  |     PyObject *ident = NULL; | 
					
						
							| 
									
										
										
										
											2022-08-06 09:55:24 +03:00
										 |  |  |     PyObject *items = NULL; | 
					
						
							|  |  |  |     PyObject *key, *value; | 
					
						
							|  |  |  |     bool first = true; | 
					
						
							| 
									
										
										
										
											2009-05-02 12:36:44 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-01-13 08:38:15 +02:00
										 |  |  |     if (PyDict_GET_SIZE(dct) == 0)  /* Fast path */ | 
					
						
							| 
									
										
										
										
											2022-07-27 11:43:34 +03:00
										 |  |  |         return _PyUnicodeWriter_WriteASCIIString(writer, "{}", 2); | 
					
						
							| 
									
										
										
										
											2009-05-02 12:36:44 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |     if (s->markers != Py_None) { | 
					
						
							|  |  |  |         int has_key; | 
					
						
							|  |  |  |         ident = PyLong_FromVoidPtr(dct); | 
					
						
							|  |  |  |         if (ident == NULL) | 
					
						
							|  |  |  |             goto bail; | 
					
						
							|  |  |  |         has_key = PyDict_Contains(s->markers, ident); | 
					
						
							|  |  |  |         if (has_key) { | 
					
						
							|  |  |  |             if (has_key != -1) | 
					
						
							|  |  |  |                 PyErr_SetString(PyExc_ValueError, "Circular reference detected"); | 
					
						
							|  |  |  |             goto bail; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         if (PyDict_SetItem(s->markers, ident, dct)) { | 
					
						
							|  |  |  |             goto bail; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-07-27 11:43:34 +03:00
										 |  |  |     if (_PyUnicodeWriter_WriteChar(writer, '{')) | 
					
						
							| 
									
										
										
										
											2009-05-02 12:36:44 +00:00
										 |  |  |         goto bail; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (s->indent != Py_None) { | 
					
						
							|  |  |  |         /* TODO: DOES NOT RUN */ | 
					
						
							|  |  |  |         indent_level += 1; | 
					
						
							|  |  |  |         /*
 | 
					
						
							|  |  |  |             newline_indent = '\n' + (' ' * (_indent * _current_indent_level)) | 
					
						
							|  |  |  |             separator = _item_separator + newline_indent | 
					
						
							|  |  |  |             buf += newline_indent | 
					
						
							|  |  |  |         */ | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-08-06 09:55:24 +03:00
										 |  |  |     if (s->sort_keys) { | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         items = PyDict_Items(dct); | 
					
						
							|  |  |  |         if (items == NULL || PyList_Sort(items) < 0) | 
					
						
							| 
									
										
										
										
											2009-05-27 09:58:34 +00:00
										 |  |  |             goto bail; | 
					
						
							| 
									
										
										
										
											2022-08-06 09:55:24 +03:00
										 |  |  | 
 | 
					
						
							|  |  |  |         for (Py_ssize_t  i = 0; i < PyList_GET_SIZE(items); i++) { | 
					
						
							|  |  |  |             PyObject *item = PyList_GET_ITEM(items, i); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             if (!PyTuple_Check(item) || PyTuple_GET_SIZE(item) != 2) { | 
					
						
							|  |  |  |                 PyErr_SetString(PyExc_ValueError, "items must return 2-tuples"); | 
					
						
							| 
									
										
										
										
											2009-05-02 12:36:44 +00:00
										 |  |  |                 goto bail; | 
					
						
							| 
									
										
										
										
											2013-08-10 13:01:45 -07:00
										 |  |  |             } | 
					
						
							| 
									
										
										
										
											2009-05-02 12:36:44 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-08-06 09:55:24 +03:00
										 |  |  |             key = PyTuple_GET_ITEM(item, 0); | 
					
						
							|  |  |  |             value = PyTuple_GET_ITEM(item, 1); | 
					
						
							|  |  |  |             if (encoder_encode_key_value(s, writer, &first, key, value, indent_level) < 0) | 
					
						
							| 
									
										
										
										
											2009-05-02 12:36:44 +00:00
										 |  |  |                 goto bail; | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											2022-08-06 09:55:24 +03:00
										 |  |  |         Py_CLEAR(items); | 
					
						
							| 
									
										
										
										
											2009-05-02 12:36:44 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-08-06 09:55:24 +03:00
										 |  |  |     } else { | 
					
						
							|  |  |  |         Py_ssize_t pos = 0; | 
					
						
							|  |  |  |         while (PyDict_Next(dct, &pos, &key, &value)) { | 
					
						
							|  |  |  |             if (encoder_encode_key_value(s, writer, &first, key, value, indent_level) < 0) | 
					
						
							|  |  |  |                 goto bail; | 
					
						
							| 
									
										
										
										
											2009-05-02 12:36:44 +00:00
										 |  |  |         } | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2009-05-27 06:50:31 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-05-02 12:36:44 +00:00
										 |  |  |     if (ident != NULL) { | 
					
						
							|  |  |  |         if (PyDict_DelItem(s->markers, ident)) | 
					
						
							|  |  |  |             goto bail; | 
					
						
							|  |  |  |         Py_CLEAR(ident); | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2011-02-22 20:15:44 +00:00
										 |  |  |     /* TODO DOES NOT RUN; dead code
 | 
					
						
							| 
									
										
										
										
											2009-05-02 12:36:44 +00:00
										 |  |  |     if (s->indent != Py_None) { | 
					
						
							|  |  |  |         indent_level -= 1; | 
					
						
							| 
									
										
										
										
											2011-02-22 20:15:44 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |         yield '\n' + (' ' * (_indent * _current_indent_level)) | 
					
						
							|  |  |  |     }*/ | 
					
						
							| 
									
										
										
										
											2022-07-27 11:43:34 +03:00
										 |  |  |     if (_PyUnicodeWriter_WriteChar(writer, '}')) | 
					
						
							| 
									
										
										
										
											2009-05-02 12:36:44 +00:00
										 |  |  |         goto bail; | 
					
						
							|  |  |  |     return 0; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | bail: | 
					
						
							| 
									
										
										
										
											2022-08-06 09:55:24 +03:00
										 |  |  |     Py_XDECREF(items); | 
					
						
							| 
									
										
										
										
											2009-05-02 12:36:44 +00:00
										 |  |  |     Py_XDECREF(ident); | 
					
						
							|  |  |  |     return -1; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static int | 
					
						
							| 
									
										
										
										
											2022-07-27 11:43:34 +03:00
										 |  |  | encoder_listencode_list(PyEncoderObject *s, _PyUnicodeWriter *writer, | 
					
						
							| 
									
										
										
										
											2011-08-19 18:03:14 +02:00
										 |  |  |                         PyObject *seq, Py_ssize_t indent_level) | 
					
						
							| 
									
										
										
										
											2009-05-02 12:36:44 +00:00
										 |  |  | { | 
					
						
							|  |  |  |     PyObject *ident = NULL; | 
					
						
							|  |  |  |     PyObject *s_fast = NULL; | 
					
						
							|  |  |  |     Py_ssize_t i; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     ident = NULL; | 
					
						
							|  |  |  |     s_fast = PySequence_Fast(seq, "_iterencode_list needs a sequence"); | 
					
						
							|  |  |  |     if (s_fast == NULL) | 
					
						
							|  |  |  |         return -1; | 
					
						
							| 
									
										
										
										
											2012-11-01 19:52:06 +01:00
										 |  |  |     if (PySequence_Fast_GET_SIZE(s_fast) == 0) { | 
					
						
							| 
									
										
										
										
											2009-05-02 12:36:44 +00:00
										 |  |  |         Py_DECREF(s_fast); | 
					
						
							| 
									
										
										
										
											2022-07-27 11:43:34 +03:00
										 |  |  |         return _PyUnicodeWriter_WriteASCIIString(writer, "[]", 2); | 
					
						
							| 
									
										
										
										
											2009-05-02 12:36:44 +00:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (s->markers != Py_None) { | 
					
						
							|  |  |  |         int has_key; | 
					
						
							|  |  |  |         ident = PyLong_FromVoidPtr(seq); | 
					
						
							|  |  |  |         if (ident == NULL) | 
					
						
							|  |  |  |             goto bail; | 
					
						
							|  |  |  |         has_key = PyDict_Contains(s->markers, ident); | 
					
						
							|  |  |  |         if (has_key) { | 
					
						
							|  |  |  |             if (has_key != -1) | 
					
						
							|  |  |  |                 PyErr_SetString(PyExc_ValueError, "Circular reference detected"); | 
					
						
							|  |  |  |             goto bail; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         if (PyDict_SetItem(s->markers, ident, seq)) { | 
					
						
							|  |  |  |             goto bail; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-07-27 11:43:34 +03:00
										 |  |  |     if (_PyUnicodeWriter_WriteChar(writer, '[')) | 
					
						
							| 
									
										
										
										
											2009-05-02 12:36:44 +00:00
										 |  |  |         goto bail; | 
					
						
							|  |  |  |     if (s->indent != Py_None) { | 
					
						
							|  |  |  |         /* TODO: DOES NOT RUN */ | 
					
						
							|  |  |  |         indent_level += 1; | 
					
						
							|  |  |  |         /*
 | 
					
						
							|  |  |  |             newline_indent = '\n' + (' ' * (_indent * _current_indent_level)) | 
					
						
							|  |  |  |             separator = _item_separator + newline_indent | 
					
						
							|  |  |  |             buf += newline_indent | 
					
						
							|  |  |  |         */ | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2012-11-01 19:52:06 +01:00
										 |  |  |     for (i = 0; i < PySequence_Fast_GET_SIZE(s_fast); i++) { | 
					
						
							|  |  |  |         PyObject *obj = PySequence_Fast_GET_ITEM(s_fast, i); | 
					
						
							| 
									
										
										
										
											2009-05-02 12:36:44 +00:00
										 |  |  |         if (i) { | 
					
						
							| 
									
										
										
										
											2022-07-27 11:43:34 +03:00
										 |  |  |             if (_PyUnicodeWriter_WriteStr(writer, s->item_separator)) | 
					
						
							| 
									
										
										
										
											2009-05-02 12:36:44 +00:00
										 |  |  |                 goto bail; | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											2022-07-27 11:43:34 +03:00
										 |  |  |         if (encoder_listencode_obj(s, writer, obj, indent_level)) | 
					
						
							| 
									
										
										
										
											2009-05-02 12:36:44 +00:00
										 |  |  |             goto bail; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     if (ident != NULL) { | 
					
						
							|  |  |  |         if (PyDict_DelItem(s->markers, ident)) | 
					
						
							|  |  |  |             goto bail; | 
					
						
							|  |  |  |         Py_CLEAR(ident); | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2011-02-22 20:15:44 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |     /* TODO: DOES NOT RUN
 | 
					
						
							| 
									
										
										
										
											2009-05-02 12:36:44 +00:00
										 |  |  |     if (s->indent != Py_None) { | 
					
						
							|  |  |  |         indent_level -= 1; | 
					
						
							| 
									
										
										
										
											2011-02-22 20:15:44 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |         yield '\n' + (' ' * (_indent * _current_indent_level)) | 
					
						
							|  |  |  |     }*/ | 
					
						
							| 
									
										
										
										
											2022-07-27 11:43:34 +03:00
										 |  |  |     if (_PyUnicodeWriter_WriteChar(writer, ']')) | 
					
						
							| 
									
										
										
										
											2009-05-02 12:36:44 +00:00
										 |  |  |         goto bail; | 
					
						
							|  |  |  |     Py_DECREF(s_fast); | 
					
						
							|  |  |  |     return 0; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | bail: | 
					
						
							|  |  |  |     Py_XDECREF(ident); | 
					
						
							|  |  |  |     Py_DECREF(s_fast); | 
					
						
							|  |  |  |     return -1; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static void | 
					
						
							|  |  |  | encoder_dealloc(PyObject *self) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2020-03-27 19:59:59 +09:00
										 |  |  |     PyTypeObject *tp = Py_TYPE(self); | 
					
						
							| 
									
										
										
										
											2017-08-24 14:55:17 +09:00
										 |  |  |     /* bpo-31095: UnTrack is needed before calling any callbacks */ | 
					
						
							|  |  |  |     PyObject_GC_UnTrack(self); | 
					
						
							| 
									
										
										
										
											2020-03-27 19:59:59 +09:00
										 |  |  |     encoder_clear((PyEncoderObject *)self); | 
					
						
							|  |  |  |     tp->tp_free(self); | 
					
						
							|  |  |  |     Py_DECREF(tp); | 
					
						
							| 
									
										
										
										
											2009-05-02 12:36:44 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static int | 
					
						
							| 
									
										
										
										
											2020-03-27 19:59:59 +09:00
										 |  |  | encoder_traverse(PyEncoderObject *self, visitproc visit, void *arg) | 
					
						
							| 
									
										
										
										
											2009-05-02 12:36:44 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2020-05-27 10:03:38 +01:00
										 |  |  |     Py_VISIT(Py_TYPE(self)); | 
					
						
							| 
									
										
										
										
											2020-03-27 19:59:59 +09:00
										 |  |  |     Py_VISIT(self->markers); | 
					
						
							|  |  |  |     Py_VISIT(self->defaultfn); | 
					
						
							|  |  |  |     Py_VISIT(self->encoder); | 
					
						
							|  |  |  |     Py_VISIT(self->indent); | 
					
						
							|  |  |  |     Py_VISIT(self->key_separator); | 
					
						
							|  |  |  |     Py_VISIT(self->item_separator); | 
					
						
							| 
									
										
										
										
											2009-05-02 12:36:44 +00:00
										 |  |  |     return 0; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static int | 
					
						
							| 
									
										
										
										
											2020-03-27 19:59:59 +09:00
										 |  |  | encoder_clear(PyEncoderObject *self) | 
					
						
							| 
									
										
										
										
											2009-05-02 12:36:44 +00:00
										 |  |  | { | 
					
						
							|  |  |  |     /* Deallocate Encoder */ | 
					
						
							| 
									
										
										
										
											2020-03-27 19:59:59 +09:00
										 |  |  |     Py_CLEAR(self->markers); | 
					
						
							|  |  |  |     Py_CLEAR(self->defaultfn); | 
					
						
							|  |  |  |     Py_CLEAR(self->encoder); | 
					
						
							|  |  |  |     Py_CLEAR(self->indent); | 
					
						
							|  |  |  |     Py_CLEAR(self->key_separator); | 
					
						
							|  |  |  |     Py_CLEAR(self->item_separator); | 
					
						
							| 
									
										
										
										
											2009-05-02 12:36:44 +00:00
										 |  |  |     return 0; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | PyDoc_STRVAR(encoder_doc, "_iterencode(obj, _current_indent_level) -> iterable"); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-03-27 19:59:59 +09:00
										 |  |  | static PyType_Slot PyEncoderType_slots[] = { | 
					
						
							|  |  |  |     {Py_tp_doc, (void *)encoder_doc}, | 
					
						
							|  |  |  |     {Py_tp_dealloc, encoder_dealloc}, | 
					
						
							|  |  |  |     {Py_tp_call, encoder_call}, | 
					
						
							|  |  |  |     {Py_tp_traverse, encoder_traverse}, | 
					
						
							|  |  |  |     {Py_tp_clear, encoder_clear}, | 
					
						
							|  |  |  |     {Py_tp_members, encoder_members}, | 
					
						
							|  |  |  |     {Py_tp_new, encoder_new}, | 
					
						
							|  |  |  |     {0, 0} | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static PyType_Spec PyEncoderType_spec = { | 
					
						
							|  |  |  |     .name = "_json.Encoder", | 
					
						
							|  |  |  |     .basicsize = sizeof(PyEncoderObject), | 
					
						
							|  |  |  |     .itemsize = 0, | 
					
						
							|  |  |  |     .flags = Py_TPFLAGS_DEFAULT | Py_TPFLAGS_HAVE_GC, | 
					
						
							|  |  |  |     .slots = PyEncoderType_slots | 
					
						
							| 
									
										
										
										
											2009-05-02 12:36:44 +00:00
										 |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static PyMethodDef speedups_methods[] = { | 
					
						
							|  |  |  |     {"encode_basestring_ascii", | 
					
						
							|  |  |  |         (PyCFunction)py_encode_basestring_ascii, | 
					
						
							|  |  |  |         METH_O, | 
					
						
							|  |  |  |         pydoc_encode_basestring_ascii}, | 
					
						
							| 
									
										
										
										
											2015-01-11 16:41:01 +01:00
										 |  |  |     {"encode_basestring", | 
					
						
							|  |  |  |         (PyCFunction)py_encode_basestring, | 
					
						
							|  |  |  |         METH_O, | 
					
						
							|  |  |  |         pydoc_encode_basestring}, | 
					
						
							| 
									
										
										
										
											2009-05-02 12:36:44 +00:00
										 |  |  |     {"scanstring", | 
					
						
							|  |  |  |         (PyCFunction)py_scanstring, | 
					
						
							|  |  |  |         METH_VARARGS, | 
					
						
							|  |  |  |         pydoc_scanstring}, | 
					
						
							|  |  |  |     {NULL, NULL, 0, NULL} | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | PyDoc_STRVAR(module_doc, | 
					
						
							|  |  |  | "json speedups\n"); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-01-16 00:32:51 +08:00
										 |  |  | static int | 
					
						
							|  |  |  | _json_exec(PyObject *module) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2022-07-02 08:24:19 +02:00
										 |  |  |     PyObject *PyScannerType = PyType_FromSpec(&PyScannerType_spec); | 
					
						
							|  |  |  |     if (PyScannerType == NULL) { | 
					
						
							| 
									
										
										
										
											2020-01-16 00:32:51 +08:00
										 |  |  |         return -1; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2022-07-02 08:24:19 +02:00
										 |  |  |     int rc = PyModule_AddObjectRef(module, "make_scanner", PyScannerType); | 
					
						
							|  |  |  |     Py_DECREF(PyScannerType); | 
					
						
							|  |  |  |     if (rc < 0) { | 
					
						
							| 
									
										
										
										
											2020-01-16 00:32:51 +08:00
										 |  |  |         return -1; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2020-03-27 19:59:59 +09:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-07-02 08:24:19 +02:00
										 |  |  |     PyObject *PyEncoderType = PyType_FromSpec(&PyEncoderType_spec); | 
					
						
							|  |  |  |     if (PyEncoderType == NULL) { | 
					
						
							| 
									
										
										
										
											2020-01-16 00:32:51 +08:00
										 |  |  |         return -1; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2022-07-02 08:24:19 +02:00
										 |  |  |     rc = PyModule_AddObjectRef(module, "make_encoder", PyEncoderType); | 
					
						
							|  |  |  |     Py_DECREF(PyEncoderType); | 
					
						
							|  |  |  |     if (rc < 0) { | 
					
						
							| 
									
										
										
										
											2020-01-16 00:32:51 +08:00
										 |  |  |         return -1; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2020-03-27 19:59:59 +09:00
										 |  |  | 
 | 
					
						
							|  |  |  |     return 0; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-01-16 00:32:51 +08:00
										 |  |  | static PyModuleDef_Slot _json_slots[] = { | 
					
						
							|  |  |  |     {Py_mod_exec, _json_exec}, | 
					
						
							|  |  |  |     {0, NULL} | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-05-02 12:36:44 +00:00
										 |  |  | static struct PyModuleDef jsonmodule = { | 
					
						
							| 
									
										
										
										
											2022-07-02 08:24:19 +02:00
										 |  |  |     .m_base = PyModuleDef_HEAD_INIT, | 
					
						
							|  |  |  |     .m_name = "_json", | 
					
						
							|  |  |  |     .m_doc = module_doc, | 
					
						
							|  |  |  |     .m_methods = speedups_methods, | 
					
						
							|  |  |  |     .m_slots = _json_slots, | 
					
						
							| 
									
										
										
										
											2009-05-02 12:36:44 +00:00
										 |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-03-17 17:48:27 +01:00
										 |  |  | PyMODINIT_FUNC | 
					
						
							| 
									
										
										
										
											2009-05-02 12:36:44 +00:00
										 |  |  | PyInit__json(void) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2020-01-16 00:32:51 +08:00
										 |  |  |     return PyModuleDef_Init(&jsonmodule); | 
					
						
							| 
									
										
										
										
											2008-05-08 14:29:10 +00:00
										 |  |  | } |