| 
									
										
										
										
											2008-05-08 14:29:10 +00:00
										 |  |  | #include "Python.h"
 | 
					
						
							| 
									
										
										
										
											2009-05-02 12:36:44 +00:00
										 |  |  | #include "structmember.h"
 | 
					
						
							|  |  |  | #if PY_VERSION_HEX < 0x02060000 && !defined(Py_TYPE)
 | 
					
						
							|  |  |  | #define Py_TYPE(ob)     (((PyObject*)(ob))->ob_type)
 | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  | #if PY_VERSION_HEX < 0x02050000 && !defined(PY_SSIZE_T_MIN)
 | 
					
						
							|  |  |  | typedef int Py_ssize_t; | 
					
						
							|  |  |  | #define PY_SSIZE_T_MAX INT_MAX
 | 
					
						
							|  |  |  | #define PY_SSIZE_T_MIN INT_MIN
 | 
					
						
							|  |  |  | #define PyInt_FromSsize_t PyInt_FromLong
 | 
					
						
							|  |  |  | #define PyInt_AsSsize_t PyInt_AsLong
 | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  | #ifndef Py_IS_FINITE
 | 
					
						
							|  |  |  | #define Py_IS_FINITE(X) (!Py_IS_INFINITY(X) && !Py_IS_NAN(X))
 | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #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 | 
					
						
							|  |  |  |     PyObject *strict; | 
					
						
							|  |  |  |     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[] = { | 
					
						
							|  |  |  |     {"strict", T_OBJECT, 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; | 
					
						
							|  |  |  |     PyObject *sort_keys; | 
					
						
							|  |  |  |     PyObject *skipkeys; | 
					
						
							|  |  |  |     int fast_encode; | 
					
						
							|  |  |  |     int allow_nan; | 
					
						
							|  |  |  | } 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_OBJECT, offsetof(PyEncoderObject, sort_keys), READONLY, "sort_keys"}, | 
					
						
							|  |  |  |     {"skipkeys", T_OBJECT, offsetof(PyEncoderObject, skipkeys), READONLY, "skipkeys"}, | 
					
						
							|  |  |  |     {NULL} | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 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 int | 
					
						
							|  |  |  | scanner_init(PyObject *self, 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 int | 
					
						
							|  |  |  | encoder_init(PyObject *self, PyObject *args, PyObject *kwds); | 
					
						
							|  |  |  | static void | 
					
						
							|  |  |  | encoder_dealloc(PyObject *self); | 
					
						
							|  |  |  | static int | 
					
						
							|  |  |  | encoder_clear(PyObject *self); | 
					
						
							|  |  |  | static int | 
					
						
							|  |  |  | encoder_listencode_list(PyEncoderObject *s, PyObject *rval, PyObject *seq, Py_ssize_t indent_level); | 
					
						
							|  |  |  | static int | 
					
						
							|  |  |  | encoder_listencode_obj(PyEncoderObject *s, PyObject *rval, PyObject *obj, Py_ssize_t indent_level); | 
					
						
							|  |  |  | static int | 
					
						
							|  |  |  | encoder_listencode_dict(PyEncoderObject *s, PyObject *rval, PyObject *dct, Py_ssize_t indent_level); | 
					
						
							|  |  |  | static PyObject * | 
					
						
							| 
									
										
										
										
											2009-05-02 15:55:19 +00:00
										 |  |  | _encoded_const(PyObject *obj); | 
					
						
							| 
									
										
										
										
											2009-05-02 12:36:44 +00:00
										 |  |  | static void | 
					
						
							|  |  |  | raise_errmsg(char *msg, PyObject *s, Py_ssize_t end); | 
					
						
							|  |  |  | static PyObject * | 
					
						
							|  |  |  | encoder_encode_string(PyEncoderObject *s, PyObject *obj); | 
					
						
							|  |  |  | static int | 
					
						
							|  |  |  | _convertPyInt_AsSsize_t(PyObject *o, Py_ssize_t *size_ptr); | 
					
						
							|  |  |  | static PyObject * | 
					
						
							|  |  |  | _convertPyInt_FromSsize_t(Py_ssize_t *size_ptr); | 
					
						
							|  |  |  | 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
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-05-02 12:36:44 +00:00
										 |  |  | #define MIN_EXPANSION 6
 | 
					
						
							| 
									
										
										
										
											2008-05-08 14:29:10 +00:00
										 |  |  | #ifdef Py_UNICODE_WIDE
 | 
					
						
							|  |  |  | #define MAX_EXPANSION (2 * MIN_EXPANSION)
 | 
					
						
							|  |  |  | #else
 | 
					
						
							|  |  |  | #define MAX_EXPANSION MIN_EXPANSION
 | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-05-02 12:36:44 +00:00
										 |  |  | static int | 
					
						
							|  |  |  | _convertPyInt_AsSsize_t(PyObject *o, Py_ssize_t *size_ptr) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     /* PyObject to Py_ssize_t converter */ | 
					
						
							|  |  |  |     *size_ptr = PyLong_AsSsize_t(o); | 
					
						
							| 
									
										
										
										
											2009-05-05 07:52:05 +00:00
										 |  |  |     if (*size_ptr == -1 && PyErr_Occurred()) | 
					
						
							|  |  |  |         return 0; | 
					
						
							|  |  |  |     return 1; | 
					
						
							| 
									
										
										
										
											2009-05-02 12:36:44 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static PyObject * | 
					
						
							|  |  |  | _convertPyInt_FromSsize_t(Py_ssize_t *size_ptr) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     /* Py_ssize_t to PyObject converter */ | 
					
						
							|  |  |  |     return PyLong_FromSsize_t(*size_ptr); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-05-08 14:29:10 +00:00
										 |  |  | static Py_ssize_t | 
					
						
							| 
									
										
										
										
											2009-05-02 12:36:44 +00:00
										 |  |  | ascii_escape_unichar(Py_UNICODE c, Py_UNICODE *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: | 
					
						
							|  |  |  | #ifdef Py_UNICODE_WIDE
 | 
					
						
							|  |  |  |             if (c >= 0x10000) { | 
					
						
							|  |  |  |                 /* UTF-16 surrogate pair */ | 
					
						
							|  |  |  |                 Py_UNICODE v = c - 0x10000; | 
					
						
							|  |  |  |                 c = 0xd800 | ((v >> 10) & 0x3ff); | 
					
						
							|  |  |  |                 output[chars++] = 'u'; | 
					
						
							| 
									
										
										
										
											2009-05-02 12:36:44 +00:00
										 |  |  |                 output[chars++] = "0123456789abcdef"[(c >> 12) & 0xf]; | 
					
						
							|  |  |  |                 output[chars++] = "0123456789abcdef"[(c >>  8) & 0xf]; | 
					
						
							|  |  |  |                 output[chars++] = "0123456789abcdef"[(c >>  4) & 0xf]; | 
					
						
							|  |  |  |                 output[chars++] = "0123456789abcdef"[(c      ) & 0xf]; | 
					
						
							| 
									
										
										
										
											2008-05-08 14:29:10 +00:00
										 |  |  |                 c = 0xdc00 | (v & 0x3ff); | 
					
						
							|  |  |  |                 output[chars++] = '\\'; | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  |             output[chars++] = 'u'; | 
					
						
							| 
									
										
										
										
											2009-05-02 12:36:44 +00:00
										 |  |  |             output[chars++] = "0123456789abcdef"[(c >> 12) & 0xf]; | 
					
						
							|  |  |  |             output[chars++] = "0123456789abcdef"[(c >>  8) & 0xf]; | 
					
						
							|  |  |  |             output[chars++] = "0123456789abcdef"[(c >>  4) & 0xf]; | 
					
						
							|  |  |  |             output[chars++] = "0123456789abcdef"[(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; | 
					
						
							| 
									
										
										
										
											2009-05-02 12:36:44 +00:00
										 |  |  |     Py_ssize_t max_output_size; | 
					
						
							| 
									
										
										
										
											2008-05-08 14:29:10 +00:00
										 |  |  |     Py_ssize_t chars; | 
					
						
							|  |  |  |     PyObject *rval; | 
					
						
							| 
									
										
										
										
											2009-05-02 12:36:44 +00:00
										 |  |  |     Py_UNICODE *output; | 
					
						
							| 
									
										
										
										
											2008-05-08 14:29:10 +00:00
										 |  |  |     Py_UNICODE *input_unicode; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     input_chars = PyUnicode_GET_SIZE(pystr); | 
					
						
							|  |  |  |     input_unicode = PyUnicode_AS_UNICODE(pystr); | 
					
						
							| 
									
										
										
										
											2009-05-02 12:36:44 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-05-08 14:29:10 +00:00
										 |  |  |     /* One char input can be up to 6 chars output, estimate 4 of these */ | 
					
						
							|  |  |  |     output_size = 2 + (MIN_EXPANSION * 4) + input_chars; | 
					
						
							| 
									
										
										
										
											2009-05-02 12:36:44 +00:00
										 |  |  |     max_output_size = 2 + (input_chars * MAX_EXPANSION); | 
					
						
							|  |  |  |     rval = PyUnicode_FromStringAndSize(NULL, output_size); | 
					
						
							| 
									
										
										
										
											2008-05-08 14:29:10 +00:00
										 |  |  |     if (rval == NULL) { | 
					
						
							|  |  |  |         return NULL; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2009-05-02 12:36:44 +00:00
										 |  |  |     output = PyUnicode_AS_UNICODE(rval); | 
					
						
							| 
									
										
										
										
											2008-05-08 14:29:10 +00:00
										 |  |  |     chars = 0; | 
					
						
							|  |  |  |     output[chars++] = '"'; | 
					
						
							|  |  |  |     for (i = 0; i < input_chars; i++) { | 
					
						
							|  |  |  |         Py_UNICODE c = input_unicode[i]; | 
					
						
							|  |  |  |         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
										 |  |  |         } | 
					
						
							|  |  |  |         if (output_size - chars < (1 + MAX_EXPANSION)) { | 
					
						
							|  |  |  |             /* There's more than four, so let's resize by a lot */ | 
					
						
							| 
									
										
										
										
											2009-05-02 12:36:44 +00:00
										 |  |  |             Py_ssize_t new_output_size = output_size * 2; | 
					
						
							| 
									
										
										
										
											2008-05-08 14:29:10 +00:00
										 |  |  |             /* This is an upper bound */ | 
					
						
							| 
									
										
										
										
											2009-05-02 12:36:44 +00:00
										 |  |  |             if (new_output_size > max_output_size) { | 
					
						
							|  |  |  |                 new_output_size = max_output_size; | 
					
						
							| 
									
										
										
										
											2008-05-08 14:29:10 +00:00
										 |  |  |             } | 
					
						
							| 
									
										
										
										
											2009-05-02 12:36:44 +00:00
										 |  |  |             /* Make sure that the output size changed before resizing */ | 
					
						
							|  |  |  |             if (new_output_size != output_size) { | 
					
						
							|  |  |  |                 output_size = new_output_size; | 
					
						
							|  |  |  |                 if (PyUnicode_Resize(&rval, output_size) == -1) { | 
					
						
							|  |  |  |                     return NULL; | 
					
						
							|  |  |  |                 } | 
					
						
							|  |  |  |                 output = PyUnicode_AS_UNICODE(rval); | 
					
						
							| 
									
										
										
										
											2008-05-08 14:29:10 +00:00
										 |  |  |             } | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     output[chars++] = '"'; | 
					
						
							| 
									
										
										
										
											2009-05-02 12:36:44 +00:00
										 |  |  |     if (PyUnicode_Resize(&rval, chars) == -1) { | 
					
						
							| 
									
										
										
										
											2008-05-08 14:29:10 +00:00
										 |  |  |         return NULL; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     return rval; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-05-02 12:36:44 +00:00
										 |  |  | static void | 
					
						
							| 
									
										
										
										
											2008-05-08 14:29:10 +00:00
										 |  |  | raise_errmsg(char *msg, PyObject *s, Py_ssize_t end) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2009-05-02 12:36:44 +00:00
										 |  |  |     /* Use the Python function json.decoder.errmsg to raise a nice
 | 
					
						
							|  |  |  |     looking ValueError exception */ | 
					
						
							| 
									
										
										
										
											2008-05-08 14:29:10 +00:00
										 |  |  |     static PyObject *errmsg_fn = NULL; | 
					
						
							|  |  |  |     PyObject *pymsg; | 
					
						
							|  |  |  |     if (errmsg_fn == NULL) { | 
					
						
							|  |  |  |         PyObject *decoder = PyImport_ImportModule("json.decoder"); | 
					
						
							|  |  |  |         if (decoder == NULL) | 
					
						
							|  |  |  |             return; | 
					
						
							|  |  |  |         errmsg_fn = PyObject_GetAttrString(decoder, "errmsg"); | 
					
						
							| 
									
										
										
										
											2009-05-02 12:36:44 +00:00
										 |  |  |         Py_DECREF(decoder); | 
					
						
							| 
									
										
										
										
											2008-05-08 14:29:10 +00:00
										 |  |  |         if (errmsg_fn == NULL) | 
					
						
							|  |  |  |             return; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2009-05-02 12:36:44 +00:00
										 |  |  |     pymsg = PyObject_CallFunction(errmsg_fn, "(zOO&)", msg, s, _convertPyInt_FromSsize_t, &end); | 
					
						
							| 
									
										
										
										
											2008-10-16 21:17:24 +00:00
										 |  |  |     if (pymsg) { | 
					
						
							|  |  |  |         PyErr_SetObject(PyExc_ValueError, pymsg); | 
					
						
							|  |  |  |         Py_DECREF(pymsg); | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2008-05-08 14:29:10 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static PyObject * | 
					
						
							|  |  |  | join_list_unicode(PyObject *lst) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2009-05-02 12:36:44 +00:00
										 |  |  |     /* return u''.join(lst) */ | 
					
						
							|  |  |  |     static PyObject *sep = NULL; | 
					
						
							|  |  |  |     if (sep == NULL) { | 
					
						
							|  |  |  |         sep = PyUnicode_FromStringAndSize("", 0); | 
					
						
							|  |  |  |         if (sep == NULL) | 
					
						
							|  |  |  |             return NULL; | 
					
						
							| 
									
										
										
										
											2008-05-08 14:29:10 +00:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2009-05-02 12:36:44 +00:00
										 |  |  |     return PyUnicode_Join(sep, lst); | 
					
						
							| 
									
										
										
										
											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
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-09-04 20:16:53 +00:00
										 |  |  | #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_DECREF(chunk); \ | 
					
						
							|  |  |  |             goto bail; \ | 
					
						
							|  |  |  |         } \ | 
					
						
							|  |  |  |         Py_CLEAR(chunk); \ | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											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; | 
					
						
							| 
									
										
										
										
											2008-05-08 14:29:10 +00:00
										 |  |  |     Py_ssize_t len = PyUnicode_GET_SIZE(pystr); | 
					
						
							|  |  |  |     Py_ssize_t begin = end - 1; | 
					
						
							|  |  |  |     Py_ssize_t next = begin; | 
					
						
							|  |  |  |     const Py_UNICODE *buf = PyUnicode_AS_UNICODE(pystr); | 
					
						
							| 
									
										
										
										
											2010-09-04 20:16:53 +00:00
										 |  |  |     PyObject *chunks = NULL; | 
					
						
							|  |  |  |     PyObject *chunk = NULL; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-07-19 22:26:35 +00:00
										 |  |  |     if (end < 0 || len <= end) { | 
					
						
							|  |  |  |         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 */ | 
					
						
							|  |  |  |         Py_UNICODE c = 0; | 
					
						
							|  |  |  |         for (next = end; next < len; next++) { | 
					
						
							|  |  |  |             c = buf[next]; | 
					
						
							|  |  |  |             if (c == '"' || c == '\\') { | 
					
						
							|  |  |  |                 break; | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |             else if (strict && c <= 0x1f) { | 
					
						
							| 
									
										
										
										
											2008-07-19 22:26:35 +00:00
										 |  |  |                 raise_errmsg("Invalid control character at", pystr, next); | 
					
						
							| 
									
										
										
										
											2008-05-08 14:29:10 +00:00
										 |  |  |                 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) { | 
					
						
							| 
									
										
										
										
											2010-09-04 20:16:53 +00:00
										 |  |  |             APPEND_OLD_CHUNK | 
					
						
							| 
									
										
										
										
											2008-05-08 14:29:10 +00:00
										 |  |  |             chunk = PyUnicode_FromUnicode(&buf[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 = 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_UNICODE digit = 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; | 
					
						
							|  |  |  |                 } | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  | #ifdef Py_UNICODE_WIDE
 | 
					
						
							|  |  |  |             /* Surrogate pair */ | 
					
						
							| 
									
										
										
										
											2009-05-02 12:36:44 +00:00
										 |  |  |             if ((c & 0xfc00) == 0xd800) { | 
					
						
							| 
									
										
										
										
											2008-05-08 14:29:10 +00:00
										 |  |  |                 Py_UNICODE c2 = 0; | 
					
						
							|  |  |  |                 if (end + 6 >= len) { | 
					
						
							| 
									
										
										
										
											2009-05-02 12:36:44 +00:00
										 |  |  |                     raise_errmsg("Unpaired high surrogate", pystr, end - 5); | 
					
						
							|  |  |  |                     goto bail; | 
					
						
							| 
									
										
										
										
											2008-05-08 14:29:10 +00:00
										 |  |  |                 } | 
					
						
							|  |  |  |                 if (buf[next++] != '\\' || buf[next++] != 'u') { | 
					
						
							| 
									
										
										
										
											2009-05-02 12:36:44 +00:00
										 |  |  |                     raise_errmsg("Unpaired high surrogate", pystr, end - 5); | 
					
						
							|  |  |  |                     goto bail; | 
					
						
							| 
									
										
										
										
											2008-05-08 14:29:10 +00:00
										 |  |  |                 } | 
					
						
							|  |  |  |                 end += 6; | 
					
						
							|  |  |  |                 /* Decode 4 hex digits */ | 
					
						
							|  |  |  |                 for (; next < end; next++) { | 
					
						
							|  |  |  |                     Py_UNICODE digit = 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; | 
					
						
							|  |  |  |                     } | 
					
						
							|  |  |  |                 } | 
					
						
							| 
									
										
										
										
											2009-05-02 12:36:44 +00:00
										 |  |  |                 if ((c2 & 0xfc00) != 0xdc00) { | 
					
						
							|  |  |  |                     raise_errmsg("Unpaired high surrogate", pystr, end - 5); | 
					
						
							|  |  |  |                     goto bail; | 
					
						
							|  |  |  |                 } | 
					
						
							| 
									
										
										
										
											2008-05-08 14:29:10 +00:00
										 |  |  |                 c = 0x10000 + (((c - 0xd800) << 10) | (c2 - 0xdc00)); | 
					
						
							|  |  |  |             } | 
					
						
							| 
									
										
										
										
											2009-05-02 12:36:44 +00:00
										 |  |  |             else if ((c & 0xfc00) == 0xdc00) { | 
					
						
							|  |  |  |                 raise_errmsg("Unpaired low surrogate", pystr, end - 5); | 
					
						
							|  |  |  |                 goto bail; | 
					
						
							|  |  |  |             } | 
					
						
							| 
									
										
										
										
											2008-05-08 14:29:10 +00:00
										 |  |  | #endif
 | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											2010-09-04 20:16:53 +00:00
										 |  |  |         APPEND_OLD_CHUNK | 
					
						
							| 
									
										
										
										
											2008-05-08 14:29:10 +00:00
										 |  |  |         chunk = PyUnicode_FromUnicode(&c, 1); | 
					
						
							|  |  |  |         if (chunk == NULL) { | 
					
						
							|  |  |  |             goto bail; | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											2010-09-04 20:16:53 +00:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     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) { | 
					
						
							| 
									
										
										
										
											2008-05-08 14:29:10 +00:00
										 |  |  |             goto bail; | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											2010-09-04 20:16:53 +00:00
										 |  |  |         Py_CLEAR(chunks); | 
					
						
							| 
									
										
										
										
											2008-05-08 14:29:10 +00:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-05-02 12:36:44 +00:00
										 |  |  |     *next_end_ptr = end; | 
					
						
							|  |  |  |     return rval; | 
					
						
							| 
									
										
										
										
											2008-05-08 14:29:10 +00:00
										 |  |  | bail: | 
					
						
							| 
									
										
										
										
											2009-05-02 12:36:44 +00:00
										 |  |  |     *next_end_ptr = -1; | 
					
						
							| 
									
										
										
										
											2008-05-08 14:29:10 +00:00
										 |  |  |     Py_XDECREF(chunks); | 
					
						
							| 
									
										
										
										
											2010-09-04 20:16:53 +00:00
										 |  |  |     Py_XDECREF(chunk); | 
					
						
							| 
									
										
										
										
											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 * | 
					
						
							| 
									
										
										
										
											2009-05-02 12:36:44 +00:00
										 |  |  | py_scanstring(PyObject* self UNUSED, 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; | 
					
						
							|  |  |  |     if (!PyArg_ParseTuple(args, "OO&|i:scanstring", &pystr, _convertPyInt_AsSsize_t, &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 * | 
					
						
							| 
									
										
										
										
											2009-05-02 12:36:44 +00:00
										 |  |  | py_encode_basestring_ascii(PyObject* self UNUSED, 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; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-05-02 12:36:44 +00:00
										 |  |  | static void | 
					
						
							|  |  |  | scanner_dealloc(PyObject *self) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     /* Deallocate scanner object */ | 
					
						
							|  |  |  |     scanner_clear(self); | 
					
						
							|  |  |  |     Py_TYPE(self)->tp_free(self); | 
					
						
							|  |  |  | } | 
					
						
							| 
									
										
										
										
											2008-05-08 14:29:10 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-05-02 12:36:44 +00:00
										 |  |  | static int | 
					
						
							|  |  |  | scanner_traverse(PyObject *self, visitproc visit, void *arg) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     PyScannerObject *s; | 
					
						
							|  |  |  |     assert(PyScanner_Check(self)); | 
					
						
							|  |  |  |     s = (PyScannerObject *)self; | 
					
						
							|  |  |  |     Py_VISIT(s->strict); | 
					
						
							|  |  |  |     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; | 
					
						
							|  |  |  | } | 
					
						
							| 
									
										
										
										
											2008-06-11 05:26:20 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-05-02 12:36:44 +00:00
										 |  |  | static int | 
					
						
							|  |  |  | scanner_clear(PyObject *self) | 
					
						
							| 
									
										
										
										
											2008-05-08 14:29:10 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2009-05-02 12:36:44 +00:00
										 |  |  |     PyScannerObject *s; | 
					
						
							|  |  |  |     assert(PyScanner_Check(self)); | 
					
						
							|  |  |  |     s = (PyScannerObject *)self; | 
					
						
							|  |  |  |     Py_CLEAR(s->strict); | 
					
						
							|  |  |  |     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); | 
					
						
							| 
									
										
										
										
											2010-09-04 20:16:53 +00:00
										 |  |  |     Py_CLEAR(s->memo); | 
					
						
							| 
									
										
										
										
											2009-05-02 12:36:44 +00:00
										 |  |  |     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) | 
					
						
							|  |  |  |     */ | 
					
						
							|  |  |  |     Py_UNICODE *str = PyUnicode_AS_UNICODE(pystr); | 
					
						
							|  |  |  |     Py_ssize_t end_idx = PyUnicode_GET_SIZE(pystr) - 1; | 
					
						
							|  |  |  |     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; | 
					
						
							|  |  |  |     int strict = PyObject_IsTrue(s->strict); | 
					
						
							| 
									
										
										
										
											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
										 |  |  | 
 | 
					
						
							|  |  |  |     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 { */ | 
					
						
							|  |  |  |     while (idx <= end_idx && IS_WHITESPACE(str[idx])) idx++; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     /* only loop if the object is non-empty */ | 
					
						
							|  |  |  |     if (idx <= end_idx && str[idx] != '}') { | 
					
						
							|  |  |  |         while (idx <= end_idx) { | 
					
						
							| 
									
										
										
										
											2010-09-04 20:16:53 +00:00
										 |  |  |             PyObject *memokey; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-05-02 12:36:44 +00:00
										 |  |  |             /* read key */ | 
					
						
							|  |  |  |             if (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; | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |             key = scanstring_unicode(pystr, idx + 1, strict, &next_idx); | 
					
						
							|  |  |  |             if (key == NULL) | 
					
						
							|  |  |  |                 goto bail; | 
					
						
							| 
									
										
										
										
											2010-09-04 20:16:53 +00:00
										 |  |  |             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; | 
					
						
							|  |  |  |             } | 
					
						
							| 
									
										
										
										
											2009-05-02 12:36:44 +00:00
										 |  |  |             idx = next_idx; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             /* skip whitespace between key and : delimiter, read :, skip whitespace */ | 
					
						
							|  |  |  |             while (idx <= end_idx && IS_WHITESPACE(str[idx])) idx++; | 
					
						
							|  |  |  |             if (idx > end_idx || 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++; | 
					
						
							|  |  |  |             while (idx <= end_idx && IS_WHITESPACE(str[idx])) idx++; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             /* 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 , */ | 
					
						
							|  |  |  |             while (idx <= end_idx && IS_WHITESPACE(str[idx])) idx++; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             /* bail if the object is closed or we didn't get the , delimiter */ | 
					
						
							|  |  |  |             if (idx > end_idx) break; | 
					
						
							|  |  |  |             if (str[idx] == '}') { | 
					
						
							|  |  |  |                 break; | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |             else if (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 */ | 
					
						
							|  |  |  |             while (idx <= end_idx && IS_WHITESPACE(str[idx])) idx++; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     /* verify that idx < end_idx, str[idx] should be '}' */ | 
					
						
							|  |  |  |     if (idx > end_idx || str[idx] != '}') { | 
					
						
							|  |  |  |         raise_errmsg("Expecting object", pystr, end_idx); | 
					
						
							|  |  |  |         goto bail; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     *next_idx_ptr = idx + 1; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-09-04 20:16:53 +00:00
										 |  |  |     if (has_pairs_hook) { | 
					
						
							| 
									
										
										
										
											2009-05-02 12:36:44 +00:00
										 |  |  |         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); | 
					
						
							| 
									
										
										
										
											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) { | 
					
						
							|  |  |  |     /* Read a JSON array from PyString 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 | 
					
						
							|  |  |  |     */ | 
					
						
							|  |  |  |     Py_UNICODE *str = PyUnicode_AS_UNICODE(pystr); | 
					
						
							|  |  |  |     Py_ssize_t end_idx = PyUnicode_GET_SIZE(pystr) - 1; | 
					
						
							|  |  |  |     PyObject *val = NULL; | 
					
						
							|  |  |  |     PyObject *rval = PyList_New(0); | 
					
						
							|  |  |  |     Py_ssize_t next_idx; | 
					
						
							|  |  |  |     if (rval == NULL) | 
					
						
							|  |  |  |         return NULL; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     /* skip whitespace after [ */ | 
					
						
							|  |  |  |     while (idx <= end_idx && IS_WHITESPACE(str[idx])) idx++; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     /* only loop if the array is non-empty */ | 
					
						
							|  |  |  |     if (idx <= end_idx && str[idx] != ']') { | 
					
						
							|  |  |  |         while (idx <= end_idx) { | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             /* 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(str[idx])) idx++; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             /* bail if the array is closed or we didn't get the , delimiter */ | 
					
						
							|  |  |  |             if (idx > end_idx) break; | 
					
						
							|  |  |  |             if (str[idx] == ']') { | 
					
						
							|  |  |  |                 break; | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |             else if (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 , */ | 
					
						
							|  |  |  |             while (idx <= end_idx && IS_WHITESPACE(str[idx])) idx++; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     /* verify that idx < end_idx, str[idx] should be ']' */ | 
					
						
							|  |  |  |     if (idx > end_idx || str[idx] != ']') { | 
					
						
							|  |  |  |         raise_errmsg("Expecting object", 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, char *constant, Py_ssize_t idx, Py_ssize_t *next_idx_ptr) { | 
					
						
							|  |  |  |     /* Read a JSON constant from PyString pystr.
 | 
					
						
							|  |  |  |     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_SIZE(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: | 
					
						
							|  |  |  |         PyInt, PyLong, or PyFloat. | 
					
						
							|  |  |  |         May return other types if parse_int or parse_float are set | 
					
						
							|  |  |  |     */ | 
					
						
							|  |  |  |     Py_UNICODE *str = PyUnicode_AS_UNICODE(pystr); | 
					
						
							|  |  |  |     Py_ssize_t end_idx = PyUnicode_GET_SIZE(pystr) - 1; | 
					
						
							|  |  |  |     Py_ssize_t idx = start; | 
					
						
							|  |  |  |     int is_float = 0; | 
					
						
							|  |  |  |     PyObject *rval; | 
					
						
							|  |  |  |     PyObject *numstr; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     /* read a sign if it's there, make sure it's not the end of the string */ | 
					
						
							|  |  |  |     if (str[idx] == '-') { | 
					
						
							|  |  |  |         idx++; | 
					
						
							|  |  |  |         if (idx > end_idx) { | 
					
						
							|  |  |  |             PyErr_SetNone(PyExc_StopIteration); | 
					
						
							|  |  |  |             return NULL; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     /* read as many integer digits as we find as long as it doesn't start with 0 */ | 
					
						
							|  |  |  |     if (str[idx] >= '1' && str[idx] <= '9') { | 
					
						
							|  |  |  |         idx++; | 
					
						
							|  |  |  |         while (idx <= end_idx && str[idx] >= '0' && str[idx] <= '9') idx++; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     /* if it starts with 0 we only expect one integer digit */ | 
					
						
							|  |  |  |     else if (str[idx] == '0') { | 
					
						
							|  |  |  |         idx++; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     /* no integer digits, error */ | 
					
						
							|  |  |  |     else { | 
					
						
							|  |  |  |         PyErr_SetNone(PyExc_StopIteration); | 
					
						
							|  |  |  |         return NULL; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     /* if the next char is '.' followed by a digit then read all float digits */ | 
					
						
							|  |  |  |     if (idx < end_idx && str[idx] == '.' && str[idx + 1] >= '0' && str[idx + 1] <= '9') { | 
					
						
							|  |  |  |         is_float = 1; | 
					
						
							|  |  |  |         idx += 2; | 
					
						
							|  |  |  |         while (idx <= end_idx && str[idx] >= '0' && str[idx] <= '9') idx++; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     /* if the next char is 'e' or 'E' then maybe read the exponent (or backtrack) */ | 
					
						
							|  |  |  |     if (idx < end_idx && (str[idx] == 'e' || str[idx] == 'E')) { | 
					
						
							|  |  |  |         Py_ssize_t e_start = idx; | 
					
						
							|  |  |  |         idx++; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         /* read an exponent sign if present */ | 
					
						
							|  |  |  |         if (idx < end_idx && (str[idx] == '-' || str[idx] == '+')) idx++; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         /* read all digits */ | 
					
						
							|  |  |  |         while (idx <= end_idx && str[idx] >= '0' && str[idx] <= '9') idx++; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         /* if we got a digit, then parse as float. if not, backtrack */ | 
					
						
							|  |  |  |         if (str[idx - 1] >= '0' && str[idx - 1] <= '9') { | 
					
						
							|  |  |  |             is_float = 1; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         else { | 
					
						
							|  |  |  |             idx = e_start; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     /* copy the section we determined to be a number */ | 
					
						
							|  |  |  |     numstr = PyUnicode_FromUnicode(&str[start], idx - start); | 
					
						
							|  |  |  |     if (numstr == NULL) | 
					
						
							|  |  |  |         return NULL; | 
					
						
							|  |  |  |     if (is_float) { | 
					
						
							|  |  |  |         /* parse as a float using a fast path if available, otherwise call user defined method */ | 
					
						
							|  |  |  |         if (s->parse_float != (PyObject *)&PyFloat_Type) { | 
					
						
							|  |  |  |             rval = PyObject_CallFunctionObjArgs(s->parse_float, numstr, NULL); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         else { | 
					
						
							|  |  |  |             rval = PyFloat_FromString(numstr); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     else { | 
					
						
							|  |  |  |         /* no fast path for unicode -> int, just call */ | 
					
						
							|  |  |  |         rval = PyObject_CallFunctionObjArgs(s->parse_int, numstr, NULL); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     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; | 
					
						
							| 
									
										
										
										
											2009-05-02 12:36:44 +00:00
										 |  |  |     Py_UNICODE *str = PyUnicode_AS_UNICODE(pystr); | 
					
						
							|  |  |  |     Py_ssize_t length = PyUnicode_GET_SIZE(pystr); | 
					
						
							|  |  |  |     if (idx >= length) { | 
					
						
							|  |  |  |         PyErr_SetNone(PyExc_StopIteration); | 
					
						
							|  |  |  |         return NULL; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     switch (str[idx]) { | 
					
						
							|  |  |  |         case '"': | 
					
						
							|  |  |  |             /* string */ | 
					
						
							|  |  |  |             return scanstring_unicode(pystr, idx + 1, | 
					
						
							|  |  |  |                 PyObject_IsTrue(s->strict), | 
					
						
							|  |  |  |                 next_idx_ptr); | 
					
						
							|  |  |  |         case '{': | 
					
						
							|  |  |  |             /* object */ | 
					
						
							| 
									
										
										
										
											2011-05-07 17:58:09 +03:00
										 |  |  |             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; | 
					
						
							| 
									
										
										
										
											2009-05-02 12:36:44 +00:00
										 |  |  |         case '[': | 
					
						
							|  |  |  |             /* array */ | 
					
						
							| 
									
										
										
										
											2011-05-07 17:58:09 +03:00
										 |  |  |             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; | 
					
						
							| 
									
										
										
										
											2009-05-02 12:36:44 +00:00
										 |  |  |         case 'n': | 
					
						
							|  |  |  |             /* null */ | 
					
						
							|  |  |  |             if ((idx + 3 < length) && str[idx + 1] == 'u' && str[idx + 2] == 'l' && str[idx + 3] == 'l') { | 
					
						
							|  |  |  |                 Py_INCREF(Py_None); | 
					
						
							|  |  |  |                 *next_idx_ptr = idx + 4; | 
					
						
							|  |  |  |                 return Py_None; | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |             break; | 
					
						
							|  |  |  |         case 't': | 
					
						
							|  |  |  |             /* true */ | 
					
						
							|  |  |  |             if ((idx + 3 < length) && str[idx + 1] == 'r' && str[idx + 2] == 'u' && str[idx + 3] == 'e') { | 
					
						
							|  |  |  |                 Py_INCREF(Py_True); | 
					
						
							|  |  |  |                 *next_idx_ptr = idx + 4; | 
					
						
							|  |  |  |                 return Py_True; | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |             break; | 
					
						
							|  |  |  |         case 'f': | 
					
						
							|  |  |  |             /* false */ | 
					
						
							|  |  |  |             if ((idx + 4 < length) && str[idx + 1] == 'a' && str[idx + 2] == 'l' && str[idx + 3] == 's' && str[idx + 4] == 'e') { | 
					
						
							|  |  |  |                 Py_INCREF(Py_False); | 
					
						
							|  |  |  |                 *next_idx_ptr = idx + 5; | 
					
						
							|  |  |  |                 return Py_False; | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |             break; | 
					
						
							|  |  |  |         case 'N': | 
					
						
							|  |  |  |             /* NaN */ | 
					
						
							|  |  |  |             if ((idx + 2 < length) && str[idx + 1] == 'a' && str[idx + 2] == 'N') { | 
					
						
							|  |  |  |                 return _parse_constant(s, "NaN", idx, next_idx_ptr); | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |             break; | 
					
						
							|  |  |  |         case 'I': | 
					
						
							|  |  |  |             /* Infinity */ | 
					
						
							|  |  |  |             if ((idx + 7 < length) && str[idx + 1] == 'n' && str[idx + 2] == 'f' && str[idx + 3] == 'i' && str[idx + 4] == 'n' && str[idx + 5] == 'i' && str[idx + 6] == 't' && str[idx + 7] == 'y') { | 
					
						
							|  |  |  |                 return _parse_constant(s, "Infinity", idx, next_idx_ptr); | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |             break; | 
					
						
							|  |  |  |         case '-': | 
					
						
							|  |  |  |             /* -Infinity */ | 
					
						
							|  |  |  |             if ((idx + 8 < length) && str[idx + 1] == 'I' && str[idx + 2] == 'n' && str[idx + 3] == 'f' && str[idx + 4] == 'i' && str[idx + 5] == 'n' && str[idx + 6] == 'i' && str[idx + 7] == 't' && 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, "OO&:scan_once", kwlist, &pystr, _convertPyInt_AsSsize_t, &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; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2010-09-04 20:16:53 +00:00
										 |  |  |     PyDict_Clear(s->memo); | 
					
						
							|  |  |  |     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; | 
					
						
							|  |  |  |     s = (PyScannerObject *)type->tp_alloc(type, 0); | 
					
						
							|  |  |  |     if (s != NULL) { | 
					
						
							|  |  |  |         s->strict = NULL; | 
					
						
							|  |  |  |         s->object_hook = NULL; | 
					
						
							|  |  |  |         s->object_pairs_hook = NULL; | 
					
						
							|  |  |  |         s->parse_float = NULL; | 
					
						
							|  |  |  |         s->parse_int = NULL; | 
					
						
							|  |  |  |         s->parse_constant = NULL; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     return (PyObject *)s; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static int | 
					
						
							|  |  |  | scanner_init(PyObject *self, PyObject *args, PyObject *kwds) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     /* Initialize Scanner object */ | 
					
						
							|  |  |  |     PyObject *ctx; | 
					
						
							|  |  |  |     static char *kwlist[] = {"context", NULL}; | 
					
						
							|  |  |  |     PyScannerObject *s; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     assert(PyScanner_Check(self)); | 
					
						
							|  |  |  |     s = (PyScannerObject *)self; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (!PyArg_ParseTupleAndKeywords(args, kwds, "O:make_scanner", kwlist, &ctx)) | 
					
						
							|  |  |  |         return -1; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-09-04 20:16:53 +00:00
										 |  |  |     if (s->memo == NULL) { | 
					
						
							|  |  |  |         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 */ | 
					
						
							|  |  |  |     s->strict = PyObject_GetAttrString(ctx, "strict"); | 
					
						
							|  |  |  |     if (s->strict == NULL) | 
					
						
							|  |  |  |         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 0; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | bail: | 
					
						
							|  |  |  |     Py_CLEAR(s->strict); | 
					
						
							|  |  |  |     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); | 
					
						
							|  |  |  |     return -1; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 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 */ | 
					
						
							|  |  |  |     scanner_init,                    /* 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) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     PyEncoderObject *s; | 
					
						
							|  |  |  |     s = (PyEncoderObject *)type->tp_alloc(type, 0); | 
					
						
							|  |  |  |     if (s != NULL) { | 
					
						
							|  |  |  |         s->markers = NULL; | 
					
						
							|  |  |  |         s->defaultfn = NULL; | 
					
						
							|  |  |  |         s->encoder = NULL; | 
					
						
							|  |  |  |         s->indent = NULL; | 
					
						
							|  |  |  |         s->key_separator = NULL; | 
					
						
							|  |  |  |         s->item_separator = NULL; | 
					
						
							|  |  |  |         s->sort_keys = NULL; | 
					
						
							|  |  |  |         s->skipkeys = NULL; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     return (PyObject *)s; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static int | 
					
						
							|  |  |  | encoder_init(PyObject *self, PyObject *args, PyObject *kwds) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     /* initialize Encoder object */ | 
					
						
							|  |  |  |     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; | 
					
						
							|  |  |  |     PyObject *item_separator, *sort_keys, *skipkeys, *allow_nan; | 
					
						
							| 
									
										
										
										
											2009-05-02 12:36:44 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |     assert(PyEncoder_Check(self)); | 
					
						
							|  |  |  |     s = (PyEncoderObject *)self; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (!PyArg_ParseTupleAndKeywords(args, kwds, "OOOOOOOOO:make_encoder", kwlist, | 
					
						
							| 
									
										
										
										
											2009-12-08 15:57:31 +00:00
										 |  |  |         &markers, &defaultfn, &encoder, &indent, &key_separator, &item_separator, | 
					
						
							|  |  |  |         &sort_keys, &skipkeys, &allow_nan)) | 
					
						
							| 
									
										
										
										
											2009-05-02 12:36:44 +00:00
										 |  |  |         return -1; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											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; | 
					
						
							|  |  |  |     s->fast_encode = (PyCFunction_Check(s->encoder) && PyCFunction_GetFunction(s->encoder) == (PyCFunction)py_encode_basestring_ascii); | 
					
						
							|  |  |  |     s->allow_nan = PyObject_IsTrue(allow_nan); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											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); | 
					
						
							|  |  |  |     Py_INCREF(s->sort_keys); | 
					
						
							|  |  |  |     Py_INCREF(s->skipkeys); | 
					
						
							|  |  |  |     return 0; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 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; | 
					
						
							|  |  |  |     PyObject *rval; | 
					
						
							|  |  |  |     Py_ssize_t indent_level; | 
					
						
							|  |  |  |     PyEncoderObject *s; | 
					
						
							|  |  |  |     assert(PyEncoder_Check(self)); | 
					
						
							|  |  |  |     s = (PyEncoderObject *)self; | 
					
						
							|  |  |  |     if (!PyArg_ParseTupleAndKeywords(args, kwds, "OO&:_iterencode", kwlist, | 
					
						
							|  |  |  |         &obj, _convertPyInt_AsSsize_t, &indent_level)) | 
					
						
							|  |  |  |         return NULL; | 
					
						
							|  |  |  |     rval = PyList_New(0); | 
					
						
							|  |  |  |     if (rval == NULL) | 
					
						
							|  |  |  |         return NULL; | 
					
						
							|  |  |  |     if (encoder_listencode_obj(s, rval, obj, indent_level)) { | 
					
						
							|  |  |  |         Py_DECREF(rval); | 
					
						
							|  |  |  |         return NULL; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     return rval; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 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"); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     /* Use a better float format here? */ | 
					
						
							|  |  |  |     return PyObject_Repr(obj); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static PyObject * | 
					
						
							|  |  |  | encoder_encode_string(PyEncoderObject *s, PyObject *obj) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     /* Return the JSON representation of a string */ | 
					
						
							|  |  |  |     if (s->fast_encode) | 
					
						
							|  |  |  |         return py_encode_basestring_ascii(NULL, obj); | 
					
						
							|  |  |  |     else | 
					
						
							|  |  |  |         return PyObject_CallFunctionObjArgs(s->encoder, obj, NULL); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static int | 
					
						
							|  |  |  | _steal_list_append(PyObject *lst, PyObject *stolen) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     /* Append stolen and then decrement its reference count */ | 
					
						
							|  |  |  |     int rval = PyList_Append(lst, stolen); | 
					
						
							|  |  |  |     Py_DECREF(stolen); | 
					
						
							|  |  |  |     return rval; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static int | 
					
						
							|  |  |  | encoder_listencode_obj(PyEncoderObject *s, PyObject *rval, PyObject *obj, Py_ssize_t indent_level) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     /* Encode Python object obj to a JSON term, rval is a PyList */ | 
					
						
							|  |  |  |     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_list_append(rval, cstr); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     else if (PyUnicode_Check(obj)) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         PyObject *encoded = encoder_encode_string(s, obj); | 
					
						
							|  |  |  |         if (encoded == NULL) | 
					
						
							|  |  |  |             return -1; | 
					
						
							|  |  |  |         return _steal_list_append(rval, encoded); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     else if (PyLong_Check(obj)) { | 
					
						
							|  |  |  |         PyObject *encoded = PyObject_Str(obj); | 
					
						
							|  |  |  |         if (encoded == NULL) | 
					
						
							|  |  |  |             return -1; | 
					
						
							|  |  |  |         return _steal_list_append(rval, encoded); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     else if (PyFloat_Check(obj)) { | 
					
						
							|  |  |  |         PyObject *encoded = encoder_encode_float(s, obj); | 
					
						
							|  |  |  |         if (encoded == NULL) | 
					
						
							|  |  |  |             return -1; | 
					
						
							|  |  |  |         return _steal_list_append(rval, encoded); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     else if (PyList_Check(obj) || PyTuple_Check(obj)) { | 
					
						
							| 
									
										
										
										
											2011-05-11 01:02:56 +03:00
										 |  |  |         if (Py_EnterRecursiveCall(" while encoding a JSON object")) | 
					
						
							|  |  |  |             return -1; | 
					
						
							|  |  |  |         rv = encoder_listencode_list(s, rval, obj, indent_level); | 
					
						
							|  |  |  |         Py_LeaveRecursiveCall(); | 
					
						
							|  |  |  |         return rv; | 
					
						
							| 
									
										
										
										
											2009-05-02 12:36:44 +00:00
										 |  |  |     } | 
					
						
							|  |  |  |     else if (PyDict_Check(obj)) { | 
					
						
							| 
									
										
										
										
											2011-05-11 01:02:56 +03:00
										 |  |  |         if (Py_EnterRecursiveCall(" while encoding a JSON object")) | 
					
						
							|  |  |  |             return -1; | 
					
						
							|  |  |  |         rv = encoder_listencode_dict(s, rval, obj, indent_level); | 
					
						
							|  |  |  |         Py_LeaveRecursiveCall(); | 
					
						
							|  |  |  |         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; | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         newobj = PyObject_CallFunctionObjArgs(s->defaultfn, obj, NULL); | 
					
						
							|  |  |  |         if (newobj == NULL) { | 
					
						
							|  |  |  |             Py_XDECREF(ident); | 
					
						
							|  |  |  |             return -1; | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											2011-05-11 01:02:56 +03:00
										 |  |  | 
 | 
					
						
							|  |  |  |         if (Py_EnterRecursiveCall(" while encoding a JSON object")) | 
					
						
							|  |  |  |             return -1; | 
					
						
							| 
									
										
										
										
											2009-05-02 12:36:44 +00:00
										 |  |  |         rv = encoder_listencode_obj(s, rval, newobj, indent_level); | 
					
						
							| 
									
										
										
										
											2011-05-11 01:02:56 +03:00
										 |  |  |         Py_LeaveRecursiveCall(); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											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; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static int | 
					
						
							|  |  |  | encoder_listencode_dict(PyEncoderObject *s, PyObject *rval, PyObject *dct, Py_ssize_t indent_level) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     /* Encode Python dict dct a JSON term, rval is a PyList */ | 
					
						
							|  |  |  |     static PyObject *open_dict = NULL; | 
					
						
							|  |  |  |     static PyObject *close_dict = NULL; | 
					
						
							|  |  |  |     static PyObject *empty_dict = NULL; | 
					
						
							|  |  |  |     PyObject *kstr = NULL; | 
					
						
							|  |  |  |     PyObject *ident = NULL; | 
					
						
							| 
									
										
										
										
											2009-05-27 06:50:31 +00:00
										 |  |  |     PyObject *it = NULL; | 
					
						
							| 
									
										
										
										
											2009-05-27 09:58:34 +00:00
										 |  |  |     PyObject *items; | 
					
						
							|  |  |  |     PyObject *item = NULL; | 
					
						
							| 
									
										
										
										
											2009-05-02 12:36:44 +00:00
										 |  |  |     int skipkeys; | 
					
						
							|  |  |  |     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; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2009-05-27 06:50:31 +00:00
										 |  |  |     if (Py_SIZE(dct) == 0) | 
					
						
							| 
									
										
										
										
											2009-05-02 12:36:44 +00:00
										 |  |  |         return PyList_Append(rval, 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 (PyList_Append(rval, 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 | 
					
						
							|  |  |  |         */ | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-05-27 09:58:34 +00:00
										 |  |  |     if (PyObject_IsTrue(s->sort_keys)) { | 
					
						
							| 
									
										
										
										
											2010-11-04 16:51:32 +00:00
										 |  |  |         /* First sort the keys then replace them with (key, value) tuples. */ | 
					
						
							|  |  |  |         Py_ssize_t i, nitems; | 
					
						
							|  |  |  |         items = PyMapping_Keys(dct); | 
					
						
							|  |  |  |         if (items == NULL) | 
					
						
							| 
									
										
										
										
											2009-05-27 11:19:02 +00:00
										 |  |  |             goto bail; | 
					
						
							| 
									
										
										
										
											2010-11-04 16:51:32 +00:00
										 |  |  |         if (!PyList_Check(items)) { | 
					
						
							|  |  |  |             PyErr_SetString(PyExc_ValueError, "keys must return list"); | 
					
						
							| 
									
										
										
										
											2009-05-27 09:58:34 +00:00
										 |  |  |             goto bail; | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											2010-11-04 16:51:32 +00:00
										 |  |  |         if (PyList_Sort(items) < 0) | 
					
						
							|  |  |  |             goto bail; | 
					
						
							|  |  |  |         nitems = PyList_GET_SIZE(items); | 
					
						
							|  |  |  |         for (i = 0; i < nitems; i++) { | 
					
						
							|  |  |  |             PyObject *key, *value; | 
					
						
							|  |  |  |             key = PyList_GET_ITEM(items, i); | 
					
						
							|  |  |  |             value = PyDict_GetItem(dct, key); | 
					
						
							|  |  |  |             item = PyTuple_Pack(2, key, value); | 
					
						
							|  |  |  |             if (item == NULL) | 
					
						
							|  |  |  |                 goto bail; | 
					
						
							|  |  |  |             PyList_SET_ITEM(items, i, item); | 
					
						
							|  |  |  |             Py_DECREF(key); | 
					
						
							| 
									
										
										
										
											2010-05-09 15:52:27 +00:00
										 |  |  |         } | 
					
						
							| 
									
										
										
										
											2010-11-04 16:51:32 +00:00
										 |  |  |     } | 
					
						
							|  |  |  |     else { | 
					
						
							|  |  |  |         items = PyMapping_Items(dct); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     if (items == NULL) | 
					
						
							| 
									
										
										
										
											2009-05-27 11:19:02 +00:00
										 |  |  |         goto bail; | 
					
						
							| 
									
										
										
										
											2009-05-27 09:58:34 +00:00
										 |  |  |     it = PyObject_GetIter(items); | 
					
						
							| 
									
										
										
										
											2010-11-04 16:51:32 +00:00
										 |  |  |     Py_DECREF(items); | 
					
						
							|  |  |  |     if (it == NULL) | 
					
						
							| 
									
										
										
										
											2009-05-27 06:50:31 +00:00
										 |  |  |         goto bail; | 
					
						
							| 
									
										
										
										
											2009-05-02 12:36:44 +00:00
										 |  |  |     skipkeys = PyObject_IsTrue(s->skipkeys); | 
					
						
							|  |  |  |     idx = 0; | 
					
						
							| 
									
										
										
										
											2009-05-27 09:58:34 +00:00
										 |  |  |     while ((item = PyIter_Next(it)) != NULL) { | 
					
						
							|  |  |  |         PyObject *encoded, *key, *value; | 
					
						
							|  |  |  |         if (!PyTuple_Check(item) || Py_SIZE(item) != 2) { | 
					
						
							|  |  |  |             PyErr_SetString(PyExc_ValueError, "items must return 2-tuples"); | 
					
						
							|  |  |  |             goto bail; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         key = PyTuple_GET_ITEM(item, 0); | 
					
						
							| 
									
										
										
										
											2009-05-02 12:36:44 +00:00
										 |  |  |         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; | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											2009-05-27 09:58:34 +00:00
										 |  |  |         else if (key == Py_True || key == Py_False || key == Py_None) { | 
					
						
							| 
									
										
										
										
											2010-05-09 15:52:27 +00:00
										 |  |  |                         /* This must come before the PyLong_Check because
 | 
					
						
							|  |  |  |                            True and False are also 1 and 0.*/ | 
					
						
							| 
									
										
										
										
											2009-05-27 09:58:34 +00:00
										 |  |  |             kstr = _encoded_const(key); | 
					
						
							| 
									
										
										
										
											2009-05-02 12:36:44 +00:00
										 |  |  |             if (kstr == NULL) | 
					
						
							|  |  |  |                 goto bail; | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											2009-05-27 09:58:34 +00:00
										 |  |  |         else if (PyLong_Check(key)) { | 
					
						
							|  |  |  |             kstr = PyObject_Str(key); | 
					
						
							| 
									
										
										
										
											2009-05-02 12:36:44 +00:00
										 |  |  |             if (kstr == NULL) | 
					
						
							|  |  |  |                 goto bail; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         else if (skipkeys) { | 
					
						
							| 
									
										
										
										
											2009-05-27 09:58:34 +00:00
										 |  |  |             Py_DECREF(item); | 
					
						
							| 
									
										
										
										
											2009-05-02 12:36:44 +00:00
										 |  |  |             continue; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         else { | 
					
						
							|  |  |  |             /* TODO: include repr of key */ | 
					
						
							| 
									
										
										
										
											2010-07-21 12:29:04 +00:00
										 |  |  |             PyErr_SetString(PyExc_TypeError, "keys must be a string"); | 
					
						
							| 
									
										
										
										
											2009-05-02 12:36:44 +00:00
										 |  |  |             goto bail; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         if (idx) { | 
					
						
							|  |  |  |             if (PyList_Append(rval, s->item_separator)) | 
					
						
							|  |  |  |                 goto bail; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         encoded = encoder_encode_string(s, kstr); | 
					
						
							|  |  |  |         Py_CLEAR(kstr); | 
					
						
							|  |  |  |         if (encoded == NULL) | 
					
						
							|  |  |  |             goto bail; | 
					
						
							|  |  |  |         if (PyList_Append(rval, encoded)) { | 
					
						
							|  |  |  |             Py_DECREF(encoded); | 
					
						
							|  |  |  |             goto bail; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         Py_DECREF(encoded); | 
					
						
							|  |  |  |         if (PyList_Append(rval, s->key_separator)) | 
					
						
							|  |  |  |             goto bail; | 
					
						
							| 
									
										
										
										
											2009-05-27 06:50:31 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-05-27 09:58:34 +00:00
										 |  |  |         value = PyTuple_GET_ITEM(item, 1); | 
					
						
							| 
									
										
										
										
											2009-05-02 12:36:44 +00:00
										 |  |  |         if (encoder_listencode_obj(s, rval, value, indent_level)) | 
					
						
							|  |  |  |             goto bail; | 
					
						
							|  |  |  |         idx += 1; | 
					
						
							| 
									
										
										
										
											2009-05-27 09:58:34 +00:00
										 |  |  |         Py_DECREF(item); | 
					
						
							| 
									
										
										
										
											2009-05-02 12:36:44 +00:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2009-05-27 06:50:31 +00:00
										 |  |  |     if (PyErr_Occurred()) | 
					
						
							|  |  |  |         goto bail; | 
					
						
							|  |  |  |     Py_CLEAR(it); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-05-02 12:36:44 +00:00
										 |  |  |     if (ident != NULL) { | 
					
						
							|  |  |  |         if (PyDict_DelItem(s->markers, ident)) | 
					
						
							|  |  |  |             goto bail; | 
					
						
							|  |  |  |         Py_CLEAR(ident); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     if (s->indent != Py_None) { | 
					
						
							|  |  |  |         /* TODO: DOES NOT RUN */ | 
					
						
							|  |  |  |         indent_level -= 1; | 
					
						
							|  |  |  |         /*
 | 
					
						
							|  |  |  |             yield '\n' + (' ' * (_indent * _current_indent_level)) | 
					
						
							|  |  |  |         */ | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     if (PyList_Append(rval, close_dict)) | 
					
						
							|  |  |  |         goto bail; | 
					
						
							|  |  |  |     return 0; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | bail: | 
					
						
							| 
									
										
										
										
											2009-05-27 06:50:31 +00:00
										 |  |  |     Py_XDECREF(it); | 
					
						
							| 
									
										
										
										
											2009-05-27 09:58:34 +00:00
										 |  |  |     Py_XDECREF(item); | 
					
						
							| 
									
										
										
										
											2009-05-02 12:36:44 +00:00
										 |  |  |     Py_XDECREF(kstr); | 
					
						
							|  |  |  |     Py_XDECREF(ident); | 
					
						
							|  |  |  |     return -1; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static int | 
					
						
							|  |  |  | encoder_listencode_list(PyEncoderObject *s, PyObject *rval, PyObject *seq, Py_ssize_t indent_level) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     /* Encode Python list seq to a JSON term, rval is a PyList */ | 
					
						
							|  |  |  |     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 num_items; | 
					
						
							|  |  |  |     PyObject **seq_items; | 
					
						
							|  |  |  |     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; | 
					
						
							|  |  |  |     num_items = PySequence_Fast_GET_SIZE(s_fast); | 
					
						
							|  |  |  |     if (num_items == 0) { | 
					
						
							|  |  |  |         Py_DECREF(s_fast); | 
					
						
							|  |  |  |         return PyList_Append(rval, 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; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     seq_items = PySequence_Fast_ITEMS(s_fast); | 
					
						
							|  |  |  |     if (PyList_Append(rval, 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 < num_items; i++) { | 
					
						
							|  |  |  |         PyObject *obj = seq_items[i]; | 
					
						
							|  |  |  |         if (i) { | 
					
						
							|  |  |  |             if (PyList_Append(rval, s->item_separator)) | 
					
						
							|  |  |  |                 goto bail; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         if (encoder_listencode_obj(s, rval, obj, indent_level)) | 
					
						
							|  |  |  |             goto bail; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     if (ident != NULL) { | 
					
						
							|  |  |  |         if (PyDict_DelItem(s->markers, ident)) | 
					
						
							|  |  |  |             goto bail; | 
					
						
							|  |  |  |         Py_CLEAR(ident); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     if (s->indent != Py_None) { | 
					
						
							|  |  |  |         /* TODO: DOES NOT RUN */ | 
					
						
							|  |  |  |         indent_level -= 1; | 
					
						
							|  |  |  |         /*
 | 
					
						
							|  |  |  |             yield '\n' + (' ' * (_indent * _current_indent_level)) | 
					
						
							|  |  |  |         */ | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     if (PyList_Append(rval, 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) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     /* Deallocate Encoder */ | 
					
						
							|  |  |  |     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); | 
					
						
							|  |  |  |     Py_VISIT(s->sort_keys); | 
					
						
							|  |  |  |     Py_VISIT(s->skipkeys); | 
					
						
							|  |  |  |     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); | 
					
						
							|  |  |  |     Py_CLEAR(s->sort_keys); | 
					
						
							|  |  |  |     Py_CLEAR(s->skipkeys); | 
					
						
							|  |  |  |     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 */ | 
					
						
							|  |  |  |     encoder_init,         /* 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}, | 
					
						
							|  |  |  |     {"scanstring", | 
					
						
							|  |  |  |         (PyCFunction)py_scanstring, | 
					
						
							|  |  |  |         METH_VARARGS, | 
					
						
							|  |  |  |         pydoc_scanstring}, | 
					
						
							|  |  |  |     {NULL, NULL, 0, NULL} | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | PyDoc_STRVAR(module_doc, | 
					
						
							|  |  |  | "json speedups\n"); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static struct PyModuleDef jsonmodule = { | 
					
						
							| 
									
										
										
										
											2010-05-09 15:52:27 +00:00
										 |  |  |         PyModuleDef_HEAD_INIT, | 
					
						
							|  |  |  |         "_json", | 
					
						
							|  |  |  |         module_doc, | 
					
						
							|  |  |  |         -1, | 
					
						
							|  |  |  |         speedups_methods, | 
					
						
							|  |  |  |         NULL, | 
					
						
							|  |  |  |         NULL, | 
					
						
							|  |  |  |         NULL, | 
					
						
							|  |  |  |         NULL | 
					
						
							| 
									
										
										
										
											2009-05-02 12:36:44 +00:00
										 |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | PyObject* | 
					
						
							|  |  |  | PyInit__json(void) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     PyObject *m = PyModule_Create(&jsonmodule); | 
					
						
							|  |  |  |     if (!m) | 
					
						
							|  |  |  |         return NULL; | 
					
						
							|  |  |  |     PyScannerType.tp_new = PyType_GenericNew; | 
					
						
							|  |  |  |     if (PyType_Ready(&PyScannerType) < 0) | 
					
						
							|  |  |  |         goto fail; | 
					
						
							|  |  |  |     PyEncoderType.tp_new = PyType_GenericNew; | 
					
						
							|  |  |  |     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; | 
					
						
							| 
									
										
										
										
											2008-05-08 14:29:10 +00:00
										 |  |  | } |