mirror of
				https://github.com/python/cpython.git
				synced 2025-10-26 11:14:33 +00:00 
			
		
		
		
	 cfa797c068
			
		
	
	
		cfa797c068
		
			
		
	
	
	
	
		
			
			Also updated an example for default() in the module docstring. Removed quotes around type name in other error messages.
		
			
				
	
	
		
			1937 lines
		
	
	
	
		
			60 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
			
		
		
	
	
			1937 lines
		
	
	
	
		
			60 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
| 
 | |
| /* Core extension modules are built-in on some platforms (e.g. Windows). */
 | |
| #ifdef Py_BUILD_CORE
 | |
| #define Py_BUILD_CORE_BUILTIN
 | |
| #undef Py_BUILD_CORE
 | |
| #endif
 | |
| 
 | |
| #include "Python.h"
 | |
| #include "structmember.h"
 | |
| #include "accu.h"
 | |
| 
 | |
| #ifdef __GNUC__
 | |
| #define UNUSED __attribute__((__unused__))
 | |
| #else
 | |
| #define UNUSED
 | |
| #endif
 | |
| 
 | |
| #define PyScanner_Check(op) PyObject_TypeCheck(op, &PyScannerType)
 | |
| #define PyScanner_CheckExact(op) (Py_TYPE(op) == &PyScannerType)
 | |
| #define PyEncoder_Check(op) PyObject_TypeCheck(op, &PyEncoderType)
 | |
| #define PyEncoder_CheckExact(op) (Py_TYPE(op) == &PyEncoderType)
 | |
| 
 | |
| static PyTypeObject PyScannerType;
 | |
| static PyTypeObject PyEncoderType;
 | |
| 
 | |
| typedef struct _PyScannerObject {
 | |
|     PyObject_HEAD
 | |
|     signed char strict;
 | |
|     PyObject *object_hook;
 | |
|     PyObject *object_pairs_hook;
 | |
|     PyObject *parse_float;
 | |
|     PyObject *parse_int;
 | |
|     PyObject *parse_constant;
 | |
|     PyObject *memo;
 | |
| } PyScannerObject;
 | |
| 
 | |
| static PyMemberDef scanner_members[] = {
 | |
|     {"strict", T_BOOL, offsetof(PyScannerObject, strict), READONLY, "strict"},
 | |
|     {"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;
 | |
|     char sort_keys;
 | |
|     char skipkeys;
 | |
|     int allow_nan;
 | |
|     PyCFunction fast_encode;
 | |
| } 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"},
 | |
|     {"sort_keys", T_BOOL, offsetof(PyEncoderObject, sort_keys), READONLY, "sort_keys"},
 | |
|     {"skipkeys", T_BOOL, offsetof(PyEncoderObject, skipkeys), READONLY, "skipkeys"},
 | |
|     {NULL}
 | |
| };
 | |
| 
 | |
| static PyObject *
 | |
| join_list_unicode(PyObject *lst)
 | |
| {
 | |
|     /* return u''.join(lst) */
 | |
|     static PyObject *sep = NULL;
 | |
|     if (sep == NULL) {
 | |
|         sep = PyUnicode_FromStringAndSize("", 0);
 | |
|         if (sep == NULL)
 | |
|             return NULL;
 | |
|     }
 | |
|     return PyUnicode_Join(sep, lst);
 | |
| }
 | |
| 
 | |
| /* Forward decls */
 | |
| 
 | |
| static PyObject *
 | |
| ascii_escape_unicode(PyObject *pystr);
 | |
| static PyObject *
 | |
| py_encode_basestring_ascii(PyObject* self UNUSED, PyObject *pystr);
 | |
| void init_json(void);
 | |
| 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
 | |
| scanner_clear(PyObject *self);
 | |
| static PyObject *
 | |
| encoder_new(PyTypeObject *type, PyObject *args, PyObject *kwds);
 | |
| static void
 | |
| encoder_dealloc(PyObject *self);
 | |
| static int
 | |
| encoder_clear(PyObject *self);
 | |
| static int
 | |
| encoder_listencode_list(PyEncoderObject *s, _PyAccu *acc, PyObject *seq, Py_ssize_t indent_level);
 | |
| static int
 | |
| encoder_listencode_obj(PyEncoderObject *s, _PyAccu *acc, PyObject *obj, Py_ssize_t indent_level);
 | |
| static int
 | |
| encoder_listencode_dict(PyEncoderObject *s, _PyAccu *acc, PyObject *dct, Py_ssize_t indent_level);
 | |
| static PyObject *
 | |
| _encoded_const(PyObject *obj);
 | |
| static void
 | |
| raise_errmsg(const char *msg, PyObject *s, Py_ssize_t end);
 | |
| static PyObject *
 | |
| encoder_encode_string(PyEncoderObject *s, PyObject *obj);
 | |
| static PyObject *
 | |
| encoder_encode_float(PyEncoderObject *s, PyObject *obj);
 | |
| 
 | |
| #define S_CHAR(c) (c >= ' ' && c <= '~' && c != '\\' && c != '"')
 | |
| #define IS_WHITESPACE(c) (((c) == ' ') || ((c) == '\t') || ((c) == '\n') || ((c) == '\r'))
 | |
| 
 | |
| static Py_ssize_t
 | |
| ascii_escape_unichar(Py_UCS4 c, unsigned char *output, Py_ssize_t chars)
 | |
| {
 | |
|     /* 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" */
 | |
|     output[chars++] = '\\';
 | |
|     switch (c) {
 | |
|         case '\\': output[chars++] = c; break;
 | |
|         case '"': output[chars++] = c; break;
 | |
|         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 */
 | |
|                 Py_UCS4 v = Py_UNICODE_HIGH_SURROGATE(c);
 | |
|                 output[chars++] = 'u';
 | |
|                 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);
 | |
|                 output[chars++] = '\\';
 | |
|             }
 | |
|             output[chars++] = 'u';
 | |
|             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];
 | |
|     }
 | |
|     return chars;
 | |
| }
 | |
| 
 | |
| static PyObject *
 | |
| ascii_escape_unicode(PyObject *pystr)
 | |
| {
 | |
|     /* Take a PyUnicode pystr and return a new ASCII-only escaped PyUnicode */
 | |
|     Py_ssize_t i;
 | |
|     Py_ssize_t input_chars;
 | |
|     Py_ssize_t output_size;
 | |
|     Py_ssize_t chars;
 | |
|     PyObject *rval;
 | |
|     void *input;
 | |
|     unsigned char *output;
 | |
|     int kind;
 | |
| 
 | |
|     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);
 | |
|         Py_ssize_t d;
 | |
|         if (S_CHAR(c)) {
 | |
|             d = 1;
 | |
|         }
 | |
|         else {
 | |
|             switch(c) {
 | |
|             case '\\': case '"': case '\b': case '\f':
 | |
|             case '\n': case '\r': case '\t':
 | |
|                 d = 2; break;
 | |
|             default:
 | |
|                 d = c >= 0x10000 ? 12 : 6;
 | |
|             }
 | |
|         }
 | |
|         if (output_size > PY_SSIZE_T_MAX - d) {
 | |
|             PyErr_SetString(PyExc_OverflowError, "string is too long to escape");
 | |
|             return NULL;
 | |
|         }
 | |
|         output_size += d;
 | |
|     }
 | |
| 
 | |
|     rval = PyUnicode_New(output_size, 127);
 | |
|     if (rval == NULL) {
 | |
|         return NULL;
 | |
|     }
 | |
|     output = PyUnicode_1BYTE_DATA(rval);
 | |
|     chars = 0;
 | |
|     output[chars++] = '"';
 | |
|     for (i = 0; i < input_chars; i++) {
 | |
|         Py_UCS4 c = PyUnicode_READ(kind, input, i);
 | |
|         if (S_CHAR(c)) {
 | |
|             output[chars++] = c;
 | |
|         }
 | |
|         else {
 | |
|             chars = ascii_escape_unichar(c, output, chars);
 | |
|         }
 | |
|     }
 | |
|     output[chars++] = '"';
 | |
| #ifdef Py_DEBUG
 | |
|     assert(_PyUnicode_CheckConsistency(rval, 1));
 | |
| #endif
 | |
|     return rval;
 | |
| }
 | |
| 
 | |
| 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;
 | |
|     void *input;
 | |
|     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);
 | |
|         Py_ssize_t d;
 | |
|         switch (c) {
 | |
|         case '\\': case '"': case '\b': case '\f':
 | |
|         case '\n': case '\r': case '\t':
 | |
|             d = 2;
 | |
|             break;
 | |
|         default:
 | |
|             if (c <= 0x1f)
 | |
|                 d = 6;
 | |
|             else
 | |
|                 d = 1;
 | |
|         }
 | |
|         if (output_size > PY_SSIZE_T_MAX - d) {
 | |
|             PyErr_SetString(PyExc_OverflowError, "string is too long to escape");
 | |
|             return NULL;
 | |
|         }
 | |
|         output_size += d;
 | |
|     }
 | |
| 
 | |
|     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;
 | |
| }
 | |
| 
 | |
| static void
 | |
| raise_errmsg(const char *msg, PyObject *s, Py_ssize_t end)
 | |
| {
 | |
|     /* Use JSONDecodeError exception to raise a nice looking ValueError subclass */
 | |
|     static PyObject *JSONDecodeError = NULL;
 | |
|     PyObject *exc;
 | |
|     if (JSONDecodeError == NULL) {
 | |
|         PyObject *decoder = PyImport_ImportModule("json.decoder");
 | |
|         if (decoder == NULL)
 | |
|             return;
 | |
|         JSONDecodeError = PyObject_GetAttrString(decoder, "JSONDecodeError");
 | |
|         Py_DECREF(decoder);
 | |
|         if (JSONDecodeError == NULL)
 | |
|             return;
 | |
|     }
 | |
|     exc = PyObject_CallFunction(JSONDecodeError, "zOn", msg, s, end);
 | |
|     if (exc) {
 | |
|         PyErr_SetObject(JSONDecodeError, exc);
 | |
|         Py_DECREF(exc);
 | |
|     }
 | |
| }
 | |
| 
 | |
| 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);
 | |
|     }
 | |
| }
 | |
| 
 | |
| static PyObject *
 | |
| _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;
 | |
|     }
 | |
|     pyidx = PyLong_FromSsize_t(idx);
 | |
|     if (pyidx == NULL) {
 | |
|         Py_DECREF(rval);
 | |
|         return NULL;
 | |
|     }
 | |
|     tpl = PyTuple_New(2);
 | |
|     if (tpl == NULL) {
 | |
|         Py_DECREF(pyidx);
 | |
|         Py_DECREF(rval);
 | |
|         return NULL;
 | |
|     }
 | |
|     PyTuple_SET_ITEM(tpl, 0, rval);
 | |
|     PyTuple_SET_ITEM(tpl, 1, pyidx);
 | |
|     return tpl;
 | |
| }
 | |
| 
 | |
| #define APPEND_OLD_CHUNK \
 | |
|     if (chunk != NULL) { \
 | |
|         if (chunks == NULL) { \
 | |
|             chunks = PyList_New(0); \
 | |
|             if (chunks == NULL) { \
 | |
|                 goto bail; \
 | |
|             } \
 | |
|         } \
 | |
|         if (PyList_Append(chunks, chunk)) { \
 | |
|             Py_CLEAR(chunk); \
 | |
|             goto bail; \
 | |
|         } \
 | |
|         Py_CLEAR(chunk); \
 | |
|     }
 | |
| 
 | |
| static PyObject *
 | |
| scanstring_unicode(PyObject *pystr, Py_ssize_t end, int strict, Py_ssize_t *next_end_ptr)
 | |
| {
 | |
|     /* 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
 | |
|     */
 | |
|     PyObject *rval = NULL;
 | |
|     Py_ssize_t len;
 | |
|     Py_ssize_t begin = end - 1;
 | |
|     Py_ssize_t next /* = begin */;
 | |
|     const void *buf;
 | |
|     int kind;
 | |
|     PyObject *chunks = NULL;
 | |
|     PyObject *chunk = NULL;
 | |
| 
 | |
|     if (PyUnicode_READY(pystr) == -1)
 | |
|         return 0;
 | |
| 
 | |
|     len = PyUnicode_GET_LENGTH(pystr);
 | |
|     buf = PyUnicode_DATA(pystr);
 | |
|     kind = PyUnicode_KIND(pystr);
 | |
| 
 | |
|     if (end < 0 || len < end) {
 | |
|         PyErr_SetString(PyExc_ValueError, "end is out of bounds");
 | |
|         goto bail;
 | |
|     }
 | |
|     while (1) {
 | |
|         /* Find the end of the string or the next escape */
 | |
|         Py_UCS4 c = 0;
 | |
|         for (next = end; next < len; next++) {
 | |
|             c = PyUnicode_READ(kind, buf, next);
 | |
|             if (c == '"' || c == '\\') {
 | |
|                 break;
 | |
|             }
 | |
|             else if (strict && c <= 0x1f) {
 | |
|                 raise_errmsg("Invalid control character at", pystr, next);
 | |
|                 goto bail;
 | |
|             }
 | |
|         }
 | |
|         if (!(c == '"' || c == '\\')) {
 | |
|             raise_errmsg("Unterminated string starting at", pystr, begin);
 | |
|             goto bail;
 | |
|         }
 | |
|         /* Pick up this chunk if it's not zero length */
 | |
|         if (next != end) {
 | |
|             APPEND_OLD_CHUNK
 | |
|                 chunk = PyUnicode_FromKindAndData(
 | |
|                     kind,
 | |
|                     (char*)buf + kind * end,
 | |
|                     next - end);
 | |
|             if (chunk == NULL) {
 | |
|                 goto bail;
 | |
|             }
 | |
|         }
 | |
|         next++;
 | |
|         if (c == '"') {
 | |
|             end = next;
 | |
|             break;
 | |
|         }
 | |
|         if (next == len) {
 | |
|             raise_errmsg("Unterminated string starting at", pystr, begin);
 | |
|             goto bail;
 | |
|         }
 | |
|         c = PyUnicode_READ(kind, buf, next);
 | |
|         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++) {
 | |
|                 Py_UCS4 digit = PyUnicode_READ(kind, buf, next);
 | |
|                 c <<= 4;
 | |
|                 switch (digit) {
 | |
|                     case '0': case '1': case '2': case '3': case '4':
 | |
|                     case '5': case '6': case '7': case '8': case '9':
 | |
|                         c |= (digit - '0'); break;
 | |
|                     case 'a': case 'b': case 'c': case 'd': case 'e':
 | |
|                     case 'f':
 | |
|                         c |= (digit - 'a' + 10); break;
 | |
|                     case 'A': case 'B': case 'C': case 'D': case 'E':
 | |
|                     case 'F':
 | |
|                         c |= (digit - 'A' + 10); break;
 | |
|                     default:
 | |
|                         raise_errmsg("Invalid \\uXXXX escape", pystr, end - 5);
 | |
|                         goto bail;
 | |
|                 }
 | |
|             }
 | |
|             /* Surrogate pair */
 | |
|             if (Py_UNICODE_IS_HIGH_SURROGATE(c) && end + 6 < len &&
 | |
|                 PyUnicode_READ(kind, buf, next++) == '\\' &&
 | |
|                 PyUnicode_READ(kind, buf, next++) == 'u') {
 | |
|                 Py_UCS4 c2 = 0;
 | |
|                 end += 6;
 | |
|                 /* Decode 4 hex digits */
 | |
|                 for (; next < end; next++) {
 | |
|                     Py_UCS4 digit = PyUnicode_READ(kind, buf, next);
 | |
|                     c2 <<= 4;
 | |
|                     switch (digit) {
 | |
|                         case '0': case '1': case '2': case '3': case '4':
 | |
|                         case '5': case '6': case '7': case '8': case '9':
 | |
|                             c2 |= (digit - '0'); break;
 | |
|                         case 'a': case 'b': case 'c': case 'd': case 'e':
 | |
|                         case 'f':
 | |
|                             c2 |= (digit - 'a' + 10); break;
 | |
|                         case 'A': case 'B': case 'C': case 'D': case 'E':
 | |
|                         case 'F':
 | |
|                             c2 |= (digit - 'A' + 10); break;
 | |
|                         default:
 | |
|                             raise_errmsg("Invalid \\uXXXX escape", pystr, end - 5);
 | |
|                             goto bail;
 | |
|                     }
 | |
|                 }
 | |
|                 if (Py_UNICODE_IS_LOW_SURROGATE(c2))
 | |
|                     c = Py_UNICODE_JOIN_SURROGATES(c, c2);
 | |
|                 else
 | |
|                     end -= 6;
 | |
|             }
 | |
|         }
 | |
|         APPEND_OLD_CHUNK
 | |
|         chunk = PyUnicode_FromKindAndData(PyUnicode_4BYTE_KIND, &c, 1);
 | |
|         if (chunk == NULL) {
 | |
|             goto bail;
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     if (chunks == NULL) {
 | |
|         if (chunk != NULL)
 | |
|             rval = chunk;
 | |
|         else
 | |
|             rval = PyUnicode_FromStringAndSize("", 0);
 | |
|     }
 | |
|     else {
 | |
|         APPEND_OLD_CHUNK
 | |
|         rval = join_list_unicode(chunks);
 | |
|         if (rval == NULL) {
 | |
|             goto bail;
 | |
|         }
 | |
|         Py_CLEAR(chunks);
 | |
|     }
 | |
| 
 | |
|     *next_end_ptr = end;
 | |
|     return rval;
 | |
| bail:
 | |
|     *next_end_ptr = -1;
 | |
|     Py_XDECREF(chunks);
 | |
|     Py_XDECREF(chunk);
 | |
|     return NULL;
 | |
| }
 | |
| 
 | |
| PyDoc_STRVAR(pydoc_scanstring,
 | |
|     "scanstring(string, end, strict=True) -> (string, end)\n"
 | |
|     "\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."
 | |
| );
 | |
| 
 | |
| static PyObject *
 | |
| py_scanstring(PyObject* self UNUSED, PyObject *args)
 | |
| {
 | |
|     PyObject *pystr;
 | |
|     PyObject *rval;
 | |
|     Py_ssize_t end;
 | |
|     Py_ssize_t next_end = -1;
 | |
|     int strict = 1;
 | |
|     if (!PyArg_ParseTuple(args, "On|i:scanstring", &pystr, &end, &strict)) {
 | |
|         return NULL;
 | |
|     }
 | |
|     if (PyUnicode_Check(pystr)) {
 | |
|         rval = scanstring_unicode(pystr, end, strict, &next_end);
 | |
|     }
 | |
|     else {
 | |
|         PyErr_Format(PyExc_TypeError,
 | |
|                      "first argument must be a string, not %.80s",
 | |
|                      Py_TYPE(pystr)->tp_name);
 | |
|         return NULL;
 | |
|     }
 | |
|     return _build_rval_index_tuple(rval, next_end);
 | |
| }
 | |
| 
 | |
| PyDoc_STRVAR(pydoc_encode_basestring_ascii,
 | |
|     "encode_basestring_ascii(string) -> string\n"
 | |
|     "\n"
 | |
|     "Return an ASCII-only JSON representation of a Python string"
 | |
| );
 | |
| 
 | |
| static PyObject *
 | |
| py_encode_basestring_ascii(PyObject* self UNUSED, PyObject *pystr)
 | |
| {
 | |
|     PyObject *rval;
 | |
|     /* Return an ASCII-only JSON representation of a Python string */
 | |
|     /* METH_O */
 | |
|     if (PyUnicode_Check(pystr)) {
 | |
|         rval = ascii_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;
 | |
| }
 | |
| 
 | |
| 
 | |
| PyDoc_STRVAR(pydoc_encode_basestring,
 | |
|     "encode_basestring(string) -> string\n"
 | |
|     "\n"
 | |
|     "Return a JSON representation of a Python string"
 | |
| );
 | |
| 
 | |
| static PyObject *
 | |
| py_encode_basestring(PyObject* self UNUSED, PyObject *pystr)
 | |
| {
 | |
|     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;
 | |
| }
 | |
| 
 | |
| static void
 | |
| scanner_dealloc(PyObject *self)
 | |
| {
 | |
|     /* bpo-31095: UnTrack is needed before calling any callbacks */
 | |
|     PyObject_GC_UnTrack(self);
 | |
|     scanner_clear(self);
 | |
|     Py_TYPE(self)->tp_free(self);
 | |
| }
 | |
| 
 | |
| static int
 | |
| scanner_traverse(PyObject *self, visitproc visit, void *arg)
 | |
| {
 | |
|     PyScannerObject *s;
 | |
|     assert(PyScanner_Check(self));
 | |
|     s = (PyScannerObject *)self;
 | |
|     Py_VISIT(s->object_hook);
 | |
|     Py_VISIT(s->object_pairs_hook);
 | |
|     Py_VISIT(s->parse_float);
 | |
|     Py_VISIT(s->parse_int);
 | |
|     Py_VISIT(s->parse_constant);
 | |
|     return 0;
 | |
| }
 | |
| 
 | |
| static int
 | |
| scanner_clear(PyObject *self)
 | |
| {
 | |
|     PyScannerObject *s;
 | |
|     assert(PyScanner_Check(self));
 | |
|     s = (PyScannerObject *)self;
 | |
|     Py_CLEAR(s->object_hook);
 | |
|     Py_CLEAR(s->object_pairs_hook);
 | |
|     Py_CLEAR(s->parse_float);
 | |
|     Py_CLEAR(s->parse_int);
 | |
|     Py_CLEAR(s->parse_constant);
 | |
|     Py_CLEAR(s->memo);
 | |
|     return 0;
 | |
| }
 | |
| 
 | |
| static PyObject *
 | |
| _parse_object_unicode(PyScannerObject *s, PyObject *pystr, Py_ssize_t idx, Py_ssize_t *next_idx_ptr)
 | |
| {
 | |
|     /* 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)
 | |
|     */
 | |
|     void *str;
 | |
|     int kind;
 | |
|     Py_ssize_t end_idx;
 | |
|     PyObject *val = NULL;
 | |
|     PyObject *rval = NULL;
 | |
|     PyObject *key = NULL;
 | |
|     int has_pairs_hook = (s->object_pairs_hook != Py_None);
 | |
|     Py_ssize_t next_idx;
 | |
| 
 | |
|     if (PyUnicode_READY(pystr) == -1)
 | |
|         return NULL;
 | |
| 
 | |
|     str = PyUnicode_DATA(pystr);
 | |
|     kind = PyUnicode_KIND(pystr);
 | |
|     end_idx = PyUnicode_GET_LENGTH(pystr) - 1;
 | |
| 
 | |
|     if (has_pairs_hook)
 | |
|         rval = PyList_New(0);
 | |
|     else
 | |
|         rval = PyDict_New();
 | |
|     if (rval == NULL)
 | |
|         return NULL;
 | |
| 
 | |
|     /* skip whitespace after { */
 | |
|     while (idx <= end_idx && IS_WHITESPACE(PyUnicode_READ(kind,str, idx))) idx++;
 | |
| 
 | |
|     /* only loop if the object is non-empty */
 | |
|     if (idx > end_idx || PyUnicode_READ(kind, str, idx) != '}') {
 | |
|         while (1) {
 | |
|             PyObject *memokey;
 | |
| 
 | |
|             /* read key */
 | |
|             if (idx > end_idx || PyUnicode_READ(kind, str, idx) != '"') {
 | |
|                 raise_errmsg("Expecting property name enclosed in double quotes", pystr, idx);
 | |
|                 goto bail;
 | |
|             }
 | |
|             key = scanstring_unicode(pystr, idx + 1, s->strict, &next_idx);
 | |
|             if (key == NULL)
 | |
|                 goto bail;
 | |
|             memokey = PyDict_GetItem(s->memo, key);
 | |
|             if (memokey != NULL) {
 | |
|                 Py_INCREF(memokey);
 | |
|                 Py_DECREF(key);
 | |
|                 key = memokey;
 | |
|             }
 | |
|             else {
 | |
|                 if (PyDict_SetItem(s->memo, key, key) < 0)
 | |
|                     goto bail;
 | |
|             }
 | |
|             idx = next_idx;
 | |
| 
 | |
|             /* skip whitespace between key and : delimiter, read :, skip whitespace */
 | |
|             while (idx <= end_idx && IS_WHITESPACE(PyUnicode_READ(kind, str, idx))) idx++;
 | |
|             if (idx > end_idx || PyUnicode_READ(kind, str, idx) != ':') {
 | |
|                 raise_errmsg("Expecting ':' delimiter", pystr, idx);
 | |
|                 goto bail;
 | |
|             }
 | |
|             idx++;
 | |
|             while (idx <= end_idx && IS_WHITESPACE(PyUnicode_READ(kind, str, idx))) idx++;
 | |
| 
 | |
|             /* read any JSON term */
 | |
|             val = scan_once_unicode(s, pystr, idx, &next_idx);
 | |
|             if (val == NULL)
 | |
|                 goto bail;
 | |
| 
 | |
|             if (has_pairs_hook) {
 | |
|                 PyObject *item = PyTuple_Pack(2, key, val);
 | |
|                 if (item == NULL)
 | |
|                     goto bail;
 | |
|                 Py_CLEAR(key);
 | |
|                 Py_CLEAR(val);
 | |
|                 if (PyList_Append(rval, item) == -1) {
 | |
|                     Py_DECREF(item);
 | |
|                     goto bail;
 | |
|                 }
 | |
|                 Py_DECREF(item);
 | |
|             }
 | |
|             else {
 | |
|                 if (PyDict_SetItem(rval, key, val) < 0)
 | |
|                     goto bail;
 | |
|                 Py_CLEAR(key);
 | |
|                 Py_CLEAR(val);
 | |
|             }
 | |
|             idx = next_idx;
 | |
| 
 | |
|             /* skip whitespace before } or , */
 | |
|             while (idx <= end_idx && IS_WHITESPACE(PyUnicode_READ(kind, str, idx))) idx++;
 | |
| 
 | |
|             /* bail if the object is closed or we didn't get the , delimiter */
 | |
|             if (idx <= end_idx && PyUnicode_READ(kind, str, idx) == '}')
 | |
|                 break;
 | |
|             if (idx > end_idx || PyUnicode_READ(kind, str, idx) != ',') {
 | |
|                 raise_errmsg("Expecting ',' delimiter", pystr, idx);
 | |
|                 goto bail;
 | |
|             }
 | |
|             idx++;
 | |
| 
 | |
|             /* skip whitespace after , delimiter */
 | |
|             while (idx <= end_idx && IS_WHITESPACE(PyUnicode_READ(kind, str, idx))) idx++;
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     *next_idx_ptr = idx + 1;
 | |
| 
 | |
|     if (has_pairs_hook) {
 | |
|         val = PyObject_CallFunctionObjArgs(s->object_pairs_hook, rval, NULL);
 | |
|         Py_DECREF(rval);
 | |
|         return val;
 | |
|     }
 | |
| 
 | |
|     /* if object_hook is not None: rval = object_hook(rval) */
 | |
|     if (s->object_hook != Py_None) {
 | |
|         val = PyObject_CallFunctionObjArgs(s->object_hook, rval, NULL);
 | |
|         Py_DECREF(rval);
 | |
|         return val;
 | |
|     }
 | |
|     return rval;
 | |
| bail:
 | |
|     Py_XDECREF(key);
 | |
|     Py_XDECREF(val);
 | |
|     Py_XDECREF(rval);
 | |
|     return NULL;
 | |
| }
 | |
| 
 | |
| static PyObject *
 | |
| _parse_array_unicode(PyScannerObject *s, PyObject *pystr, Py_ssize_t idx, Py_ssize_t *next_idx_ptr) {
 | |
|     /* Read a JSON array from PyUnicode pystr.
 | |
|     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
 | |
|     */
 | |
|     void *str;
 | |
|     int kind;
 | |
|     Py_ssize_t end_idx;
 | |
|     PyObject *val = NULL;
 | |
|     PyObject *rval;
 | |
|     Py_ssize_t next_idx;
 | |
| 
 | |
|     if (PyUnicode_READY(pystr) == -1)
 | |
|         return NULL;
 | |
| 
 | |
|     rval = PyList_New(0);
 | |
|     if (rval == NULL)
 | |
|         return NULL;
 | |
| 
 | |
|     str = PyUnicode_DATA(pystr);
 | |
|     kind = PyUnicode_KIND(pystr);
 | |
|     end_idx = PyUnicode_GET_LENGTH(pystr) - 1;
 | |
| 
 | |
|     /* skip whitespace after [ */
 | |
|     while (idx <= end_idx && IS_WHITESPACE(PyUnicode_READ(kind, str, idx))) idx++;
 | |
| 
 | |
|     /* only loop if the array is non-empty */
 | |
|     if (idx > end_idx || PyUnicode_READ(kind, str, idx) != ']') {
 | |
|         while (1) {
 | |
| 
 | |
|             /* 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 , */
 | |
|             while (idx <= end_idx && IS_WHITESPACE(PyUnicode_READ(kind, str, idx))) idx++;
 | |
| 
 | |
|             /* bail if the array is closed or we didn't get the , delimiter */
 | |
|             if (idx <= end_idx && PyUnicode_READ(kind, str, idx) == ']')
 | |
|                 break;
 | |
|             if (idx > end_idx || PyUnicode_READ(kind, str, idx) != ',') {
 | |
|                 raise_errmsg("Expecting ',' delimiter", pystr, idx);
 | |
|                 goto bail;
 | |
|             }
 | |
|             idx++;
 | |
| 
 | |
|             /* skip whitespace after , */
 | |
|             while (idx <= end_idx && IS_WHITESPACE(PyUnicode_READ(kind, str, idx))) idx++;
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     /* verify that idx < end_idx, PyUnicode_READ(kind, str, idx) should be ']' */
 | |
|     if (idx > end_idx || PyUnicode_READ(kind, str, idx) != ']') {
 | |
|         raise_errmsg("Expecting value", pystr, end_idx);
 | |
|         goto bail;
 | |
|     }
 | |
|     *next_idx_ptr = idx + 1;
 | |
|     return rval;
 | |
| bail:
 | |
|     Py_XDECREF(val);
 | |
|     Py_DECREF(rval);
 | |
|     return NULL;
 | |
| }
 | |
| 
 | |
| static PyObject *
 | |
| _parse_constant(PyScannerObject *s, const char *constant, Py_ssize_t idx, Py_ssize_t *next_idx_ptr) {
 | |
|     /* Read a JSON constant.
 | |
|     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) */
 | |
|     rval = PyObject_CallFunctionObjArgs(s->parse_constant, cstr, NULL);
 | |
|     idx += PyUnicode_GET_LENGTH(cstr);
 | |
|     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:
 | |
|         PyLong, or PyFloat.
 | |
|         May return other types if parse_int or parse_float are set
 | |
|     */
 | |
|     void *str;
 | |
|     int kind;
 | |
|     Py_ssize_t end_idx;
 | |
|     Py_ssize_t idx = start;
 | |
|     int is_float = 0;
 | |
|     PyObject *rval;
 | |
|     PyObject *numstr = NULL;
 | |
|     PyObject *custom_func;
 | |
| 
 | |
|     if (PyUnicode_READY(pystr) == -1)
 | |
|         return NULL;
 | |
| 
 | |
|     str = PyUnicode_DATA(pystr);
 | |
|     kind = PyUnicode_KIND(pystr);
 | |
|     end_idx = PyUnicode_GET_LENGTH(pystr) - 1;
 | |
| 
 | |
|     /* read a sign if it's there, make sure it's not the end of the string */
 | |
|     if (PyUnicode_READ(kind, str, idx) == '-') {
 | |
|         idx++;
 | |
|         if (idx > end_idx) {
 | |
|             raise_stop_iteration(start);
 | |
|             return NULL;
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     /* read as many integer digits as we find as long as it doesn't start with 0 */
 | |
|     if (PyUnicode_READ(kind, str, idx) >= '1' && PyUnicode_READ(kind, str, idx) <= '9') {
 | |
|         idx++;
 | |
|         while (idx <= end_idx && PyUnicode_READ(kind, str, idx) >= '0' && PyUnicode_READ(kind, str, idx) <= '9') idx++;
 | |
|     }
 | |
|     /* if it starts with 0 we only expect one integer digit */
 | |
|     else if (PyUnicode_READ(kind, str, idx) == '0') {
 | |
|         idx++;
 | |
|     }
 | |
|     /* no integer digits, error */
 | |
|     else {
 | |
|         raise_stop_iteration(start);
 | |
|         return NULL;
 | |
|     }
 | |
| 
 | |
|     /* if the next char is '.' followed by a digit then read all float digits */
 | |
|     if (idx < end_idx && PyUnicode_READ(kind, str, idx) == '.' && PyUnicode_READ(kind, str, idx + 1) >= '0' && PyUnicode_READ(kind, str, idx + 1) <= '9') {
 | |
|         is_float = 1;
 | |
|         idx += 2;
 | |
|         while (idx <= end_idx && PyUnicode_READ(kind, str, idx) >= '0' && PyUnicode_READ(kind, str, idx) <= '9') idx++;
 | |
|     }
 | |
| 
 | |
|     /* if the next char is 'e' or 'E' then maybe read the exponent (or backtrack) */
 | |
|     if (idx < end_idx && (PyUnicode_READ(kind, str, idx) == 'e' || PyUnicode_READ(kind, str, idx) == 'E')) {
 | |
|         Py_ssize_t e_start = idx;
 | |
|         idx++;
 | |
| 
 | |
|         /* read an exponent sign if present */
 | |
|         if (idx < end_idx && (PyUnicode_READ(kind, str, idx) == '-' || PyUnicode_READ(kind, str, idx) == '+')) idx++;
 | |
| 
 | |
|         /* read all digits */
 | |
|         while (idx <= end_idx && PyUnicode_READ(kind, str, idx) >= '0' && PyUnicode_READ(kind, str, idx) <= '9') idx++;
 | |
| 
 | |
|         /* if we got a digit, then parse as float. if not, backtrack */
 | |
|         if (PyUnicode_READ(kind, str, idx - 1) >= '0' && PyUnicode_READ(kind, str, idx - 1) <= '9') {
 | |
|             is_float = 1;
 | |
|         }
 | |
|         else {
 | |
|             idx = e_start;
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     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 */
 | |
|         numstr = PyUnicode_FromKindAndData(kind,
 | |
|                                            (char*)str + kind * start,
 | |
|                                            idx - start);
 | |
|         if (numstr == NULL)
 | |
|             return NULL;
 | |
|         rval = PyObject_CallFunctionObjArgs(custom_func, numstr, NULL);
 | |
|     }
 | |
|     else {
 | |
|         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++) {
 | |
|             buf[i] = (char) PyUnicode_READ(kind, str, i + start);
 | |
|         }
 | |
|         if (is_float)
 | |
|             rval = PyFloat_FromString(numstr);
 | |
|         else
 | |
|             rval = PyLong_FromString(buf, NULL, 10);
 | |
|     }
 | |
|     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.
 | |
|     */
 | |
|     PyObject *res;
 | |
|     void *str;
 | |
|     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);
 | |
| 
 | |
|     if (idx < 0) {
 | |
|         PyErr_SetString(PyExc_ValueError, "idx cannot be negative");
 | |
|         return NULL;
 | |
|     }
 | |
|     if (idx >= length) {
 | |
|         raise_stop_iteration(idx);
 | |
|         return NULL;
 | |
|     }
 | |
| 
 | |
|     switch (PyUnicode_READ(kind, str, idx)) {
 | |
|         case '"':
 | |
|             /* string */
 | |
|             return scanstring_unicode(pystr, idx + 1, s->strict, next_idx_ptr);
 | |
|         case '{':
 | |
|             /* object */
 | |
|             if (Py_EnterRecursiveCall(" while decoding a JSON object "
 | |
|                                       "from a unicode string"))
 | |
|                 return NULL;
 | |
|             res = _parse_object_unicode(s, pystr, idx + 1, next_idx_ptr);
 | |
|             Py_LeaveRecursiveCall();
 | |
|             return res;
 | |
|         case '[':
 | |
|             /* array */
 | |
|             if (Py_EnterRecursiveCall(" while decoding a JSON array "
 | |
|                                       "from a unicode string"))
 | |
|                 return NULL;
 | |
|             res = _parse_array_unicode(s, pystr, idx + 1, next_idx_ptr);
 | |
|             Py_LeaveRecursiveCall();
 | |
|             return res;
 | |
|         case 'n':
 | |
|             /* null */
 | |
|             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') {
 | |
|                 *next_idx_ptr = idx + 4;
 | |
|                 Py_RETURN_NONE;
 | |
|             }
 | |
|             break;
 | |
|         case 't':
 | |
|             /* true */
 | |
|             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') {
 | |
|                 *next_idx_ptr = idx + 4;
 | |
|                 Py_RETURN_TRUE;
 | |
|             }
 | |
|             break;
 | |
|         case 'f':
 | |
|             /* false */
 | |
|             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' &&
 | |
|                 PyUnicode_READ(kind, str, idx + 4) == 'e') {
 | |
|                 *next_idx_ptr = idx + 5;
 | |
|                 Py_RETURN_FALSE;
 | |
|             }
 | |
|             break;
 | |
|         case 'N':
 | |
|             /* NaN */
 | |
|             if ((idx + 2 < length) && PyUnicode_READ(kind, str, idx + 1) == 'a' &&
 | |
|                 PyUnicode_READ(kind, str, idx + 2) == 'N') {
 | |
|                 return _parse_constant(s, "NaN", idx, next_idx_ptr);
 | |
|             }
 | |
|             break;
 | |
|         case 'I':
 | |
|             /* Infinity */
 | |
|             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' &&
 | |
|                 PyUnicode_READ(kind, str, idx + 4) == 'n' &&
 | |
|                 PyUnicode_READ(kind, str, idx + 5) == 'i' &&
 | |
|                 PyUnicode_READ(kind, str, idx + 6) == 't' &&
 | |
|                 PyUnicode_READ(kind, str, idx + 7) == 'y') {
 | |
|                 return _parse_constant(s, "Infinity", idx, next_idx_ptr);
 | |
|             }
 | |
|             break;
 | |
|         case '-':
 | |
|             /* -Infinity */
 | |
|             if ((idx + 8 < length) && PyUnicode_READ(kind, str, idx + 1) == 'I' &&
 | |
|                 PyUnicode_READ(kind, str, idx + 2) == 'n' &&
 | |
|                 PyUnicode_READ(kind, str, idx + 3) == 'f' &&
 | |
|                 PyUnicode_READ(kind, str, idx + 4) == 'i' &&
 | |
|                 PyUnicode_READ(kind, str, idx + 5) == 'n' &&
 | |
|                 PyUnicode_READ(kind, str, idx + 6) == 'i' &&
 | |
|                 PyUnicode_READ(kind, str, idx + 7) == 't' &&
 | |
|                 PyUnicode_READ(kind, str, idx + 8) == 'y') {
 | |
|                 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 *
 | |
| scanner_call(PyObject *self, PyObject *args, PyObject *kwds)
 | |
| {
 | |
|     /* 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};
 | |
|     PyScannerObject *s;
 | |
|     assert(PyScanner_Check(self));
 | |
|     s = (PyScannerObject *)self;
 | |
|     if (!PyArg_ParseTupleAndKeywords(args, kwds, "On:scan_once", kwlist, &pystr, &idx))
 | |
|         return NULL;
 | |
| 
 | |
|     if (PyUnicode_Check(pystr)) {
 | |
|         rval = scan_once_unicode(s, pystr, idx, &next_idx);
 | |
|     }
 | |
|     else {
 | |
|         PyErr_Format(PyExc_TypeError,
 | |
|                  "first argument must be a string, not %.80s",
 | |
|                  Py_TYPE(pystr)->tp_name);
 | |
|         return NULL;
 | |
|     }
 | |
|     PyDict_Clear(s->memo);
 | |
|     if (rval == NULL)
 | |
|         return NULL;
 | |
|     return _build_rval_index_tuple(rval, next_idx);
 | |
| }
 | |
| 
 | |
| static PyObject *
 | |
| scanner_new(PyTypeObject *type, PyObject *args, PyObject *kwds)
 | |
| {
 | |
|     PyScannerObject *s;
 | |
|     PyObject *ctx;
 | |
|     PyObject *strict;
 | |
|     static char *kwlist[] = {"context", NULL};
 | |
| 
 | |
|     if (!PyArg_ParseTupleAndKeywords(args, kwds, "O:make_scanner", kwlist, &ctx))
 | |
|         return NULL;
 | |
| 
 | |
|     s = (PyScannerObject *)type->tp_alloc(type, 0);
 | |
|     if (s == NULL) {
 | |
|         return NULL;
 | |
|     }
 | |
| 
 | |
|     s->memo = PyDict_New();
 | |
|     if (s->memo == NULL)
 | |
|         goto bail;
 | |
| 
 | |
|     /* All of these will fail "gracefully" so we don't need to verify them */
 | |
|     strict = PyObject_GetAttrString(ctx, "strict");
 | |
|     if (strict == NULL)
 | |
|         goto bail;
 | |
|     s->strict = PyObject_IsTrue(strict);
 | |
|     Py_DECREF(strict);
 | |
|     if (s->strict < 0)
 | |
|         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;
 | |
| 
 | |
|     return (PyObject *)s;
 | |
| 
 | |
| bail:
 | |
|     Py_DECREF(s);
 | |
|     return NULL;
 | |
| }
 | |
| 
 | |
| PyDoc_STRVAR(scanner_doc, "JSON scanner object");
 | |
| 
 | |
| static
 | |
| PyTypeObject PyScannerType = {
 | |
|     PyVarObject_HEAD_INIT(NULL, 0)
 | |
|     "_json.Scanner",       /* tp_name */
 | |
|     sizeof(PyScannerObject), /* tp_basicsize */
 | |
|     0,                    /* tp_itemsize */
 | |
|     scanner_dealloc, /* tp_dealloc */
 | |
|     0,                    /* tp_print */
 | |
|     0,                    /* tp_getattr */
 | |
|     0,                    /* tp_setattr */
 | |
|     0,                    /* tp_compare */
 | |
|     0,                    /* tp_repr */
 | |
|     0,                    /* tp_as_number */
 | |
|     0,                    /* tp_as_sequence */
 | |
|     0,                    /* tp_as_mapping */
 | |
|     0,                    /* tp_hash */
 | |
|     scanner_call,         /* tp_call */
 | |
|     0,                    /* tp_str */
 | |
|     0,/* PyObject_GenericGetAttr, */                    /* tp_getattro */
 | |
|     0,/* PyObject_GenericSetAttr, */                    /* tp_setattro */
 | |
|     0,                    /* tp_as_buffer */
 | |
|     Py_TPFLAGS_DEFAULT | Py_TPFLAGS_HAVE_GC,   /* tp_flags */
 | |
|     scanner_doc,          /* tp_doc */
 | |
|     scanner_traverse,                    /* tp_traverse */
 | |
|     scanner_clear,                    /* tp_clear */
 | |
|     0,                    /* tp_richcompare */
 | |
|     0,                    /* tp_weaklistoffset */
 | |
|     0,                    /* tp_iter */
 | |
|     0,                    /* tp_iternext */
 | |
|     0,                    /* tp_methods */
 | |
|     scanner_members,                    /* tp_members */
 | |
|     0,                    /* tp_getset */
 | |
|     0,                    /* tp_base */
 | |
|     0,                    /* tp_dict */
 | |
|     0,                    /* tp_descr_get */
 | |
|     0,                    /* tp_descr_set */
 | |
|     0,                    /* tp_dictoffset */
 | |
|     0,                    /* tp_init */
 | |
|     0,/* PyType_GenericAlloc, */        /* tp_alloc */
 | |
|     scanner_new,          /* tp_new */
 | |
|     0,/* PyObject_GC_Del, */              /* tp_free */
 | |
| };
 | |
| 
 | |
| 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;
 | |
|     PyObject *markers, *defaultfn, *encoder, *indent, *key_separator;
 | |
|     PyObject *item_separator;
 | |
|     int sort_keys, skipkeys, allow_nan;
 | |
| 
 | |
|     if (!PyArg_ParseTupleAndKeywords(args, kwds, "OOOOUUppp:make_encoder", kwlist,
 | |
|         &markers, &defaultfn, &encoder, &indent,
 | |
|         &key_separator, &item_separator,
 | |
|         &sort_keys, &skipkeys, &allow_nan))
 | |
|         return NULL;
 | |
| 
 | |
|     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);
 | |
|         return NULL;
 | |
|     }
 | |
| 
 | |
|     s = (PyEncoderObject *)type->tp_alloc(type, 0);
 | |
|     if (s == NULL)
 | |
|         return NULL;
 | |
| 
 | |
|     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;
 | |
|     s->allow_nan = allow_nan;
 | |
|     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;
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     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);
 | |
|     return (PyObject *)s;
 | |
| }
 | |
| 
 | |
| static PyObject *
 | |
| encoder_call(PyObject *self, PyObject *args, PyObject *kwds)
 | |
| {
 | |
|     /* Python callable interface to encode_listencode_obj */
 | |
|     static char *kwlist[] = {"obj", "_current_indent_level", NULL};
 | |
|     PyObject *obj;
 | |
|     Py_ssize_t indent_level;
 | |
|     PyEncoderObject *s;
 | |
|     _PyAccu acc;
 | |
| 
 | |
|     assert(PyEncoder_Check(self));
 | |
|     s = (PyEncoderObject *)self;
 | |
|     if (!PyArg_ParseTupleAndKeywords(args, kwds, "On:_iterencode", kwlist,
 | |
|         &obj, &indent_level))
 | |
|         return NULL;
 | |
|     if (_PyAccu_Init(&acc))
 | |
|         return NULL;
 | |
|     if (encoder_listencode_obj(s, &acc, obj, indent_level)) {
 | |
|         _PyAccu_Destroy(&acc);
 | |
|         return NULL;
 | |
|     }
 | |
|     return _PyAccu_FinishAsList(&acc);
 | |
| }
 | |
| 
 | |
| static PyObject *
 | |
| _encoded_const(PyObject *obj)
 | |
| {
 | |
|     /* Return the JSON string representation of None, True, False */
 | |
|     if (obj == Py_None) {
 | |
|         static PyObject *s_null = NULL;
 | |
|         if (s_null == NULL) {
 | |
|             s_null = PyUnicode_InternFromString("null");
 | |
|         }
 | |
|         Py_INCREF(s_null);
 | |
|         return s_null;
 | |
|     }
 | |
|     else if (obj == Py_True) {
 | |
|         static PyObject *s_true = NULL;
 | |
|         if (s_true == NULL) {
 | |
|             s_true = PyUnicode_InternFromString("true");
 | |
|         }
 | |
|         Py_INCREF(s_true);
 | |
|         return s_true;
 | |
|     }
 | |
|     else if (obj == Py_False) {
 | |
|         static PyObject *s_false = NULL;
 | |
|         if (s_false == NULL) {
 | |
|             s_false = PyUnicode_InternFromString("false");
 | |
|         }
 | |
|         Py_INCREF(s_false);
 | |
|         return s_false;
 | |
|     }
 | |
|     else {
 | |
|         PyErr_SetString(PyExc_ValueError, "not a const");
 | |
|         return NULL;
 | |
|     }
 | |
| }
 | |
| 
 | |
| static PyObject *
 | |
| encoder_encode_float(PyEncoderObject *s, PyObject *obj)
 | |
| {
 | |
|     /* Return the JSON representation of a PyFloat. */
 | |
|     double i = PyFloat_AS_DOUBLE(obj);
 | |
|     if (!Py_IS_FINITE(i)) {
 | |
|         if (!s->allow_nan) {
 | |
|             PyErr_SetString(
 | |
|                     PyExc_ValueError,
 | |
|                     "Out of range float values are not JSON compliant"
 | |
|                     );
 | |
|             return NULL;
 | |
|         }
 | |
|         if (i > 0) {
 | |
|             return PyUnicode_FromString("Infinity");
 | |
|         }
 | |
|         else if (i < 0) {
 | |
|             return PyUnicode_FromString("-Infinity");
 | |
|         }
 | |
|         else {
 | |
|             return PyUnicode_FromString("NaN");
 | |
|         }
 | |
|     }
 | |
|     return PyFloat_Type.tp_repr(obj);
 | |
| }
 | |
| 
 | |
| static PyObject *
 | |
| encoder_encode_string(PyEncoderObject *s, PyObject *obj)
 | |
| {
 | |
|     /* Return the JSON representation of a string */
 | |
|     PyObject *encoded;
 | |
| 
 | |
|     if (s->fast_encode) {
 | |
|         return s->fast_encode(NULL, obj);
 | |
|     }
 | |
|     encoded = PyObject_CallFunctionObjArgs(s->encoder, obj, NULL);
 | |
|     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;
 | |
| }
 | |
| 
 | |
| static int
 | |
| _steal_accumulate(_PyAccu *acc, PyObject *stolen)
 | |
| {
 | |
|     /* Append stolen and then decrement its reference count */
 | |
|     int rval = _PyAccu_Accumulate(acc, stolen);
 | |
|     Py_DECREF(stolen);
 | |
|     return rval;
 | |
| }
 | |
| 
 | |
| static int
 | |
| encoder_listencode_obj(PyEncoderObject *s, _PyAccu *acc,
 | |
|                        PyObject *obj, Py_ssize_t indent_level)
 | |
| {
 | |
|     /* Encode Python object obj to a JSON term */
 | |
|     PyObject *newobj;
 | |
|     int rv;
 | |
| 
 | |
|     if (obj == Py_None || obj == Py_True || obj == Py_False) {
 | |
|         PyObject *cstr = _encoded_const(obj);
 | |
|         if (cstr == NULL)
 | |
|             return -1;
 | |
|         return _steal_accumulate(acc, cstr);
 | |
|     }
 | |
|     else if (PyUnicode_Check(obj))
 | |
|     {
 | |
|         PyObject *encoded = encoder_encode_string(s, obj);
 | |
|         if (encoded == NULL)
 | |
|             return -1;
 | |
|         return _steal_accumulate(acc, encoded);
 | |
|     }
 | |
|     else if (PyLong_Check(obj)) {
 | |
|         PyObject *encoded = PyLong_Type.tp_str(obj);
 | |
|         if (encoded == NULL)
 | |
|             return -1;
 | |
|         return _steal_accumulate(acc, encoded);
 | |
|     }
 | |
|     else if (PyFloat_Check(obj)) {
 | |
|         PyObject *encoded = encoder_encode_float(s, obj);
 | |
|         if (encoded == NULL)
 | |
|             return -1;
 | |
|         return _steal_accumulate(acc, encoded);
 | |
|     }
 | |
|     else if (PyList_Check(obj) || PyTuple_Check(obj)) {
 | |
|         if (Py_EnterRecursiveCall(" while encoding a JSON object"))
 | |
|             return -1;
 | |
|         rv = encoder_listencode_list(s, acc, obj, indent_level);
 | |
|         Py_LeaveRecursiveCall();
 | |
|         return rv;
 | |
|     }
 | |
|     else if (PyDict_Check(obj)) {
 | |
|         if (Py_EnterRecursiveCall(" while encoding a JSON object"))
 | |
|             return -1;
 | |
|         rv = encoder_listencode_dict(s, acc, obj, indent_level);
 | |
|         Py_LeaveRecursiveCall();
 | |
|         return rv;
 | |
|     }
 | |
|     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;
 | |
|             }
 | |
|         }
 | |
|         newobj = PyObject_CallFunctionObjArgs(s->defaultfn, obj, NULL);
 | |
|         if (newobj == NULL) {
 | |
|             Py_XDECREF(ident);
 | |
|             return -1;
 | |
|         }
 | |
| 
 | |
|         if (Py_EnterRecursiveCall(" while encoding a JSON object")) {
 | |
|             Py_DECREF(newobj);
 | |
|             Py_XDECREF(ident);
 | |
|             return -1;
 | |
|         }
 | |
|         rv = encoder_listencode_obj(s, acc, newobj, indent_level);
 | |
|         Py_LeaveRecursiveCall();
 | |
| 
 | |
|         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;
 | |
|     }
 | |
| }
 | |
| 
 | |
| static int
 | |
| encoder_listencode_dict(PyEncoderObject *s, _PyAccu *acc,
 | |
|                         PyObject *dct, Py_ssize_t indent_level)
 | |
| {
 | |
|     /* Encode Python dict dct a JSON term */
 | |
|     static PyObject *open_dict = NULL;
 | |
|     static PyObject *close_dict = NULL;
 | |
|     static PyObject *empty_dict = NULL;
 | |
|     PyObject *kstr = NULL;
 | |
|     PyObject *ident = NULL;
 | |
|     PyObject *it = NULL;
 | |
|     PyObject *items;
 | |
|     PyObject *item = NULL;
 | |
|     Py_ssize_t idx;
 | |
| 
 | |
|     if (open_dict == NULL || close_dict == NULL || empty_dict == NULL) {
 | |
|         open_dict = PyUnicode_InternFromString("{");
 | |
|         close_dict = PyUnicode_InternFromString("}");
 | |
|         empty_dict = PyUnicode_InternFromString("{}");
 | |
|         if (open_dict == NULL || close_dict == NULL || empty_dict == NULL)
 | |
|             return -1;
 | |
|     }
 | |
|     if (PyDict_GET_SIZE(dct) == 0)  /* Fast path */
 | |
|         return _PyAccu_Accumulate(acc, empty_dict);
 | |
| 
 | |
|     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;
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     if (_PyAccu_Accumulate(acc, open_dict))
 | |
|         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
 | |
|         */
 | |
|     }
 | |
| 
 | |
|     items = PyMapping_Items(dct);
 | |
|     if (items == NULL)
 | |
|         goto bail;
 | |
|     if (s->sort_keys && PyList_Sort(items) < 0) {
 | |
|         Py_DECREF(items);
 | |
|         goto bail;
 | |
|     }
 | |
|     it = PyObject_GetIter(items);
 | |
|     Py_DECREF(items);
 | |
|     if (it == NULL)
 | |
|         goto bail;
 | |
|     idx = 0;
 | |
|     while ((item = PyIter_Next(it)) != NULL) {
 | |
|         PyObject *encoded, *key, *value;
 | |
|         if (!PyTuple_Check(item) || PyTuple_GET_SIZE(item) != 2) {
 | |
|             PyErr_SetString(PyExc_ValueError, "items must return 2-tuples");
 | |
|             goto bail;
 | |
|         }
 | |
|         key = PyTuple_GET_ITEM(item, 0);
 | |
|         if (PyUnicode_Check(key)) {
 | |
|             Py_INCREF(key);
 | |
|             kstr = key;
 | |
|         }
 | |
|         else if (PyFloat_Check(key)) {
 | |
|             kstr = encoder_encode_float(s, key);
 | |
|             if (kstr == NULL)
 | |
|                 goto bail;
 | |
|         }
 | |
|         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.*/
 | |
|             kstr = _encoded_const(key);
 | |
|             if (kstr == NULL)
 | |
|                 goto bail;
 | |
|         }
 | |
|         else if (PyLong_Check(key)) {
 | |
|             kstr = PyLong_Type.tp_str(key);
 | |
|             if (kstr == NULL) {
 | |
|                 goto bail;
 | |
|             }
 | |
|         }
 | |
|         else if (s->skipkeys) {
 | |
|             Py_DECREF(item);
 | |
|             continue;
 | |
|         }
 | |
|         else {
 | |
|             PyErr_Format(PyExc_TypeError,
 | |
|                          "keys must be str, int, float, bool or None, "
 | |
|                          "not %.100s", key->ob_type->tp_name);
 | |
|             goto bail;
 | |
|         }
 | |
| 
 | |
|         if (idx) {
 | |
|             if (_PyAccu_Accumulate(acc, s->item_separator))
 | |
|                 goto bail;
 | |
|         }
 | |
| 
 | |
|         encoded = encoder_encode_string(s, kstr);
 | |
|         Py_CLEAR(kstr);
 | |
|         if (encoded == NULL)
 | |
|             goto bail;
 | |
|         if (_PyAccu_Accumulate(acc, encoded)) {
 | |
|             Py_DECREF(encoded);
 | |
|             goto bail;
 | |
|         }
 | |
|         Py_DECREF(encoded);
 | |
|         if (_PyAccu_Accumulate(acc, s->key_separator))
 | |
|             goto bail;
 | |
| 
 | |
|         value = PyTuple_GET_ITEM(item, 1);
 | |
|         if (encoder_listencode_obj(s, acc, value, indent_level))
 | |
|             goto bail;
 | |
|         idx += 1;
 | |
|         Py_DECREF(item);
 | |
|     }
 | |
|     if (PyErr_Occurred())
 | |
|         goto bail;
 | |
|     Py_CLEAR(it);
 | |
| 
 | |
|     if (ident != NULL) {
 | |
|         if (PyDict_DelItem(s->markers, ident))
 | |
|             goto bail;
 | |
|         Py_CLEAR(ident);
 | |
|     }
 | |
|     /* TODO DOES NOT RUN; dead code
 | |
|     if (s->indent != Py_None) {
 | |
|         indent_level -= 1;
 | |
| 
 | |
|         yield '\n' + (' ' * (_indent * _current_indent_level))
 | |
|     }*/
 | |
|     if (_PyAccu_Accumulate(acc, close_dict))
 | |
|         goto bail;
 | |
|     return 0;
 | |
| 
 | |
| bail:
 | |
|     Py_XDECREF(it);
 | |
|     Py_XDECREF(item);
 | |
|     Py_XDECREF(kstr);
 | |
|     Py_XDECREF(ident);
 | |
|     return -1;
 | |
| }
 | |
| 
 | |
| 
 | |
| static int
 | |
| encoder_listencode_list(PyEncoderObject *s, _PyAccu *acc,
 | |
|                         PyObject *seq, Py_ssize_t indent_level)
 | |
| {
 | |
|     /* Encode Python list seq to a JSON term */
 | |
|     static PyObject *open_array = NULL;
 | |
|     static PyObject *close_array = NULL;
 | |
|     static PyObject *empty_array = NULL;
 | |
|     PyObject *ident = NULL;
 | |
|     PyObject *s_fast = NULL;
 | |
|     Py_ssize_t i;
 | |
| 
 | |
|     if (open_array == NULL || close_array == NULL || empty_array == NULL) {
 | |
|         open_array = PyUnicode_InternFromString("[");
 | |
|         close_array = PyUnicode_InternFromString("]");
 | |
|         empty_array = PyUnicode_InternFromString("[]");
 | |
|         if (open_array == NULL || close_array == NULL || empty_array == NULL)
 | |
|             return -1;
 | |
|     }
 | |
|     ident = NULL;
 | |
|     s_fast = PySequence_Fast(seq, "_iterencode_list needs a sequence");
 | |
|     if (s_fast == NULL)
 | |
|         return -1;
 | |
|     if (PySequence_Fast_GET_SIZE(s_fast) == 0) {
 | |
|         Py_DECREF(s_fast);
 | |
|         return _PyAccu_Accumulate(acc, empty_array);
 | |
|     }
 | |
| 
 | |
|     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;
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     if (_PyAccu_Accumulate(acc, open_array))
 | |
|         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
 | |
|         */
 | |
|     }
 | |
|     for (i = 0; i < PySequence_Fast_GET_SIZE(s_fast); i++) {
 | |
|         PyObject *obj = PySequence_Fast_GET_ITEM(s_fast, i);
 | |
|         if (i) {
 | |
|             if (_PyAccu_Accumulate(acc, s->item_separator))
 | |
|                 goto bail;
 | |
|         }
 | |
|         if (encoder_listencode_obj(s, acc, obj, indent_level))
 | |
|             goto bail;
 | |
|     }
 | |
|     if (ident != NULL) {
 | |
|         if (PyDict_DelItem(s->markers, ident))
 | |
|             goto bail;
 | |
|         Py_CLEAR(ident);
 | |
|     }
 | |
| 
 | |
|     /* TODO: DOES NOT RUN
 | |
|     if (s->indent != Py_None) {
 | |
|         indent_level -= 1;
 | |
| 
 | |
|         yield '\n' + (' ' * (_indent * _current_indent_level))
 | |
|     }*/
 | |
|     if (_PyAccu_Accumulate(acc, close_array))
 | |
|         goto bail;
 | |
|     Py_DECREF(s_fast);
 | |
|     return 0;
 | |
| 
 | |
| bail:
 | |
|     Py_XDECREF(ident);
 | |
|     Py_DECREF(s_fast);
 | |
|     return -1;
 | |
| }
 | |
| 
 | |
| static void
 | |
| encoder_dealloc(PyObject *self)
 | |
| {
 | |
|     /* bpo-31095: UnTrack is needed before calling any callbacks */
 | |
|     PyObject_GC_UnTrack(self);
 | |
|     encoder_clear(self);
 | |
|     Py_TYPE(self)->tp_free(self);
 | |
| }
 | |
| 
 | |
| static int
 | |
| encoder_traverse(PyObject *self, visitproc visit, void *arg)
 | |
| {
 | |
|     PyEncoderObject *s;
 | |
|     assert(PyEncoder_Check(self));
 | |
|     s = (PyEncoderObject *)self;
 | |
|     Py_VISIT(s->markers);
 | |
|     Py_VISIT(s->defaultfn);
 | |
|     Py_VISIT(s->encoder);
 | |
|     Py_VISIT(s->indent);
 | |
|     Py_VISIT(s->key_separator);
 | |
|     Py_VISIT(s->item_separator);
 | |
|     return 0;
 | |
| }
 | |
| 
 | |
| static int
 | |
| encoder_clear(PyObject *self)
 | |
| {
 | |
|     /* Deallocate Encoder */
 | |
|     PyEncoderObject *s;
 | |
|     assert(PyEncoder_Check(self));
 | |
|     s = (PyEncoderObject *)self;
 | |
|     Py_CLEAR(s->markers);
 | |
|     Py_CLEAR(s->defaultfn);
 | |
|     Py_CLEAR(s->encoder);
 | |
|     Py_CLEAR(s->indent);
 | |
|     Py_CLEAR(s->key_separator);
 | |
|     Py_CLEAR(s->item_separator);
 | |
|     return 0;
 | |
| }
 | |
| 
 | |
| PyDoc_STRVAR(encoder_doc, "_iterencode(obj, _current_indent_level) -> iterable");
 | |
| 
 | |
| static
 | |
| PyTypeObject PyEncoderType = {
 | |
|     PyVarObject_HEAD_INIT(NULL, 0)
 | |
|     "_json.Encoder",       /* tp_name */
 | |
|     sizeof(PyEncoderObject), /* tp_basicsize */
 | |
|     0,                    /* tp_itemsize */
 | |
|     encoder_dealloc, /* tp_dealloc */
 | |
|     0,                    /* tp_print */
 | |
|     0,                    /* tp_getattr */
 | |
|     0,                    /* tp_setattr */
 | |
|     0,                    /* tp_compare */
 | |
|     0,                    /* tp_repr */
 | |
|     0,                    /* tp_as_number */
 | |
|     0,                    /* tp_as_sequence */
 | |
|     0,                    /* tp_as_mapping */
 | |
|     0,                    /* tp_hash */
 | |
|     encoder_call,         /* tp_call */
 | |
|     0,                    /* tp_str */
 | |
|     0,                    /* tp_getattro */
 | |
|     0,                    /* tp_setattro */
 | |
|     0,                    /* tp_as_buffer */
 | |
|     Py_TPFLAGS_DEFAULT | Py_TPFLAGS_HAVE_GC,   /* tp_flags */
 | |
|     encoder_doc,          /* tp_doc */
 | |
|     encoder_traverse,     /* tp_traverse */
 | |
|     encoder_clear,        /* tp_clear */
 | |
|     0,                    /* tp_richcompare */
 | |
|     0,                    /* tp_weaklistoffset */
 | |
|     0,                    /* tp_iter */
 | |
|     0,                    /* tp_iternext */
 | |
|     0,                    /* tp_methods */
 | |
|     encoder_members,      /* tp_members */
 | |
|     0,                    /* tp_getset */
 | |
|     0,                    /* tp_base */
 | |
|     0,                    /* tp_dict */
 | |
|     0,                    /* tp_descr_get */
 | |
|     0,                    /* tp_descr_set */
 | |
|     0,                    /* tp_dictoffset */
 | |
|     0,                    /* tp_init */
 | |
|     0,                    /* tp_alloc */
 | |
|     encoder_new,          /* tp_new */
 | |
|     0,                    /* tp_free */
 | |
| };
 | |
| 
 | |
| static PyMethodDef speedups_methods[] = {
 | |
|     {"encode_basestring_ascii",
 | |
|         (PyCFunction)py_encode_basestring_ascii,
 | |
|         METH_O,
 | |
|         pydoc_encode_basestring_ascii},
 | |
|     {"encode_basestring",
 | |
|         (PyCFunction)py_encode_basestring,
 | |
|         METH_O,
 | |
|         pydoc_encode_basestring},
 | |
|     {"scanstring",
 | |
|         (PyCFunction)py_scanstring,
 | |
|         METH_VARARGS,
 | |
|         pydoc_scanstring},
 | |
|     {NULL, NULL, 0, NULL}
 | |
| };
 | |
| 
 | |
| PyDoc_STRVAR(module_doc,
 | |
| "json speedups\n");
 | |
| 
 | |
| static struct PyModuleDef jsonmodule = {
 | |
|         PyModuleDef_HEAD_INIT,
 | |
|         "_json",
 | |
|         module_doc,
 | |
|         -1,
 | |
|         speedups_methods,
 | |
|         NULL,
 | |
|         NULL,
 | |
|         NULL,
 | |
|         NULL
 | |
| };
 | |
| 
 | |
| PyMODINIT_FUNC
 | |
| PyInit__json(void)
 | |
| {
 | |
|     PyObject *m = PyModule_Create(&jsonmodule);
 | |
|     if (!m)
 | |
|         return NULL;
 | |
|     if (PyType_Ready(&PyScannerType) < 0)
 | |
|         goto fail;
 | |
|     if (PyType_Ready(&PyEncoderType) < 0)
 | |
|         goto fail;
 | |
|     Py_INCREF((PyObject*)&PyScannerType);
 | |
|     if (PyModule_AddObject(m, "make_scanner", (PyObject*)&PyScannerType) < 0) {
 | |
|         Py_DECREF((PyObject*)&PyScannerType);
 | |
|         goto fail;
 | |
|     }
 | |
|     Py_INCREF((PyObject*)&PyEncoderType);
 | |
|     if (PyModule_AddObject(m, "make_encoder", (PyObject*)&PyEncoderType) < 0) {
 | |
|         Py_DECREF((PyObject*)&PyEncoderType);
 | |
|         goto fail;
 | |
|     }
 | |
|     return m;
 | |
|   fail:
 | |
|     Py_DECREF(m);
 | |
|     return NULL;
 | |
| }
 |