| #include "Python.h" |
| #include "structmember.h" |
| #include "accu.h" |
| |
| #ifdef __GNUC__ |
| #define UNUSED __attribute__((__unused__)) |
| #else |
| #define UNUSED |
| #endif |
| |
| #define PyScanner_Check(op) PyObject_TypeCheck(op, &PyScannerType) |
| #define PyScanner_CheckExact(op) (Py_TYPE(op) == &PyScannerType) |
| #define PyEncoder_Check(op) PyObject_TypeCheck(op, &PyEncoderType) |
| #define PyEncoder_CheckExact(op) (Py_TYPE(op) == &PyEncoderType) |
| |
| static PyTypeObject PyScannerType; |
| static PyTypeObject PyEncoderType; |
| |
| typedef struct _PyScannerObject { |
| PyObject_HEAD |
| PyObject *strict; |
| PyObject *object_hook; |
| PyObject *object_pairs_hook; |
| PyObject *parse_float; |
| PyObject *parse_int; |
| PyObject *parse_constant; |
| PyObject *memo; |
| } PyScannerObject; |
| |
| static PyMemberDef scanner_members[] = { |
| {"strict", T_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; |
| PyCFunction 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 * |
| join_list_unicode(PyObject *lst) |
| { |
| /* return u''.join(lst) */ |
| static PyObject *sep = NULL; |
| if (sep == NULL) { |
| sep = PyUnicode_FromStringAndSize("", 0); |
| if (sep == NULL) |
| return NULL; |
| } |
| return PyUnicode_Join(sep, lst); |
| } |
| |
| /* Forward decls */ |
| |
| static PyObject * |
| ascii_escape_unicode(PyObject *pystr); |
| static PyObject * |
| py_encode_basestring_ascii(PyObject* self UNUSED, PyObject *pystr); |
| void init_json(void); |
| static PyObject * |
| scan_once_unicode(PyScannerObject *s, PyObject *pystr, Py_ssize_t idx, Py_ssize_t *next_idx_ptr); |
| static PyObject * |
| _build_rval_index_tuple(PyObject *rval, Py_ssize_t idx); |
| static PyObject * |
| scanner_new(PyTypeObject *type, PyObject *args, PyObject *kwds); |
| static 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, _PyAccu *acc, PyObject *seq, Py_ssize_t indent_level); |
| static int |
| encoder_listencode_obj(PyEncoderObject *s, _PyAccu *acc, PyObject *obj, Py_ssize_t indent_level); |
| static int |
| encoder_listencode_dict(PyEncoderObject *s, _PyAccu *acc, PyObject *dct, Py_ssize_t indent_level); |
| static PyObject * |
| _encoded_const(PyObject *obj); |
| static void |
| raise_errmsg(char *msg, PyObject *s, Py_ssize_t end); |
| static PyObject * |
| encoder_encode_string(PyEncoderObject *s, PyObject *obj); |
| static PyObject * |
| encoder_encode_long(PyEncoderObject* s UNUSED, PyObject *obj); |
| static PyObject * |
| encoder_encode_float(PyEncoderObject *s, PyObject *obj); |
| |
| #define S_CHAR(c) (c >= ' ' && c <= '~' && c != '\\' && c != '"') |
| #define IS_WHITESPACE(c) (((c) == ' ') || ((c) == '\t') || ((c) == '\n') || ((c) == '\r')) |
| |
| static Py_ssize_t |
| ascii_escape_unichar(Py_UCS4 c, unsigned char *output, Py_ssize_t chars) |
| { |
| /* Escape unicode code point c to ASCII escape sequences |
| in char *output. output must have at least 12 bytes unused to |
| accommodate an escaped surrogate pair "\uXXXX\uXXXX" */ |
| output[chars++] = '\\'; |
| switch (c) { |
| case '\\': output[chars++] = c; break; |
| case '"': output[chars++] = c; break; |
| case '\b': output[chars++] = 'b'; break; |
| case '\f': output[chars++] = 'f'; break; |
| case '\n': output[chars++] = 'n'; break; |
| case '\r': output[chars++] = 'r'; break; |
| case '\t': output[chars++] = 't'; break; |
| default: |
| if (c >= 0x10000) { |
| /* UTF-16 surrogate pair */ |
| Py_UCS4 v = Py_UNICODE_HIGH_SURROGATE(c); |
| output[chars++] = 'u'; |
| output[chars++] = Py_hexdigits[(v >> 12) & 0xf]; |
| output[chars++] = Py_hexdigits[(v >> 8) & 0xf]; |
| output[chars++] = Py_hexdigits[(v >> 4) & 0xf]; |
| output[chars++] = Py_hexdigits[(v ) & 0xf]; |
| c = Py_UNICODE_LOW_SURROGATE(c); |
| output[chars++] = '\\'; |
| } |
| output[chars++] = 'u'; |
| output[chars++] = Py_hexdigits[(c >> 12) & 0xf]; |
| output[chars++] = Py_hexdigits[(c >> 8) & 0xf]; |
| output[chars++] = Py_hexdigits[(c >> 4) & 0xf]; |
| output[chars++] = Py_hexdigits[(c ) & 0xf]; |
| } |
| return chars; |
| } |
| |
| static PyObject * |
| ascii_escape_unicode(PyObject *pystr) |
| { |
| /* Take a PyUnicode pystr and return a new ASCII-only escaped PyUnicode */ |
| Py_ssize_t i; |
| Py_ssize_t input_chars; |
| Py_ssize_t output_size; |
| Py_ssize_t chars; |
| PyObject *rval; |
| void *input; |
| unsigned char *output; |
| int kind; |
| |
| if (PyUnicode_READY(pystr) == -1) |
| return NULL; |
| |
| input_chars = PyUnicode_GET_LENGTH(pystr); |
| input = PyUnicode_DATA(pystr); |
| kind = PyUnicode_KIND(pystr); |
| |
| /* Compute the output size */ |
| for (i = 0, output_size = 2; i < input_chars; i++) { |
| Py_UCS4 c = PyUnicode_READ(kind, input, i); |
| Py_ssize_t d; |
| if (S_CHAR(c)) { |
| d = 1; |
| } |
| else { |
| switch(c) { |
| case '\\': case '"': case '\b': case '\f': |
| case '\n': case '\r': case '\t': |
| d = 2; break; |
| default: |
| d = c >= 0x10000 ? 12 : 6; |
| } |
| } |
| if (output_size > PY_SSIZE_T_MAX - d) { |
| PyErr_SetString(PyExc_OverflowError, "string is too long to escape"); |
| return NULL; |
| } |
| output_size += d; |
| } |
| |
| rval = PyUnicode_New(output_size, 127); |
| if (rval == NULL) { |
| return NULL; |
| } |
| output = PyUnicode_1BYTE_DATA(rval); |
| chars = 0; |
| output[chars++] = '"'; |
| for (i = 0; i < input_chars; i++) { |
| Py_UCS4 c = PyUnicode_READ(kind, input, i); |
| if (S_CHAR(c)) { |
| output[chars++] = c; |
| } |
| else { |
| chars = ascii_escape_unichar(c, output, chars); |
| } |
| } |
| output[chars++] = '"'; |
| #ifdef Py_DEBUG |
| assert(_PyUnicode_CheckConsistency(rval, 1)); |
| #endif |
| return rval; |
| } |
| |
| static PyObject * |
| escape_unicode(PyObject *pystr) |
| { |
| /* Take a PyUnicode pystr and return a new escaped PyUnicode */ |
| Py_ssize_t i; |
| Py_ssize_t input_chars; |
| Py_ssize_t output_size; |
| Py_ssize_t chars; |
| PyObject *rval; |
| void *input; |
| int kind; |
| Py_UCS4 maxchar; |
| |
| if (PyUnicode_READY(pystr) == -1) |
| return NULL; |
| |
| maxchar = PyUnicode_MAX_CHAR_VALUE(pystr); |
| input_chars = PyUnicode_GET_LENGTH(pystr); |
| input = PyUnicode_DATA(pystr); |
| kind = PyUnicode_KIND(pystr); |
| |
| /* Compute the output size */ |
| for (i = 0, output_size = 2; i < input_chars; i++) { |
| Py_UCS4 c = PyUnicode_READ(kind, input, i); |
| Py_ssize_t d; |
| switch (c) { |
| case '\\': case '"': case '\b': case '\f': |
| case '\n': case '\r': case '\t': |
| d = 2; |
| break; |
| default: |
| if (c <= 0x1f) |
| d = 6; |
| else |
| d = 1; |
| } |
| if (output_size > PY_SSIZE_T_MAX - d) { |
| PyErr_SetString(PyExc_OverflowError, "string is too long to escape"); |
| return NULL; |
| } |
| output_size += d; |
| } |
| |
| rval = PyUnicode_New(output_size, maxchar); |
| if (rval == NULL) |
| return NULL; |
| |
| kind = PyUnicode_KIND(rval); |
| |
| #define ENCODE_OUTPUT do { \ |
| chars = 0; \ |
| output[chars++] = '"'; \ |
| for (i = 0; i < input_chars; i++) { \ |
| Py_UCS4 c = PyUnicode_READ(kind, input, i); \ |
| switch (c) { \ |
| case '\\': output[chars++] = '\\'; output[chars++] = c; break; \ |
| case '"': output[chars++] = '\\'; output[chars++] = c; break; \ |
| case '\b': output[chars++] = '\\'; output[chars++] = 'b'; break; \ |
| case '\f': output[chars++] = '\\'; output[chars++] = 'f'; break; \ |
| case '\n': output[chars++] = '\\'; output[chars++] = 'n'; break; \ |
| case '\r': output[chars++] = '\\'; output[chars++] = 'r'; break; \ |
| case '\t': output[chars++] = '\\'; output[chars++] = 't'; break; \ |
| default: \ |
| if (c <= 0x1f) { \ |
| output[chars++] = '\\'; \ |
| output[chars++] = 'u'; \ |
| output[chars++] = '0'; \ |
| output[chars++] = '0'; \ |
| output[chars++] = Py_hexdigits[(c >> 4) & 0xf]; \ |
| output[chars++] = Py_hexdigits[(c ) & 0xf]; \ |
| } else { \ |
| output[chars++] = c; \ |
| } \ |
| } \ |
| } \ |
| output[chars++] = '"'; \ |
| } while (0) |
| |
| if (kind == PyUnicode_1BYTE_KIND) { |
| Py_UCS1 *output = PyUnicode_1BYTE_DATA(rval); |
| ENCODE_OUTPUT; |
| } else if (kind == PyUnicode_2BYTE_KIND) { |
| Py_UCS2 *output = PyUnicode_2BYTE_DATA(rval); |
| ENCODE_OUTPUT; |
| } else { |
| Py_UCS4 *output = PyUnicode_4BYTE_DATA(rval); |
| assert(kind == PyUnicode_4BYTE_KIND); |
| ENCODE_OUTPUT; |
| } |
| #undef ENCODE_OUTPUT |
| |
| #ifdef Py_DEBUG |
| assert(_PyUnicode_CheckConsistency(rval, 1)); |
| #endif |
| return rval; |
| } |
| |
| static void |
| raise_errmsg(char *msg, PyObject *s, Py_ssize_t end) |
| { |
| /* Use JSONDecodeError exception to raise a nice looking ValueError subclass */ |
| static PyObject *JSONDecodeError = NULL; |
| PyObject *exc; |
| if (JSONDecodeError == NULL) { |
| PyObject *decoder = PyImport_ImportModule("json.decoder"); |
| if (decoder == NULL) |
| return; |
| JSONDecodeError = PyObject_GetAttrString(decoder, "JSONDecodeError"); |
| Py_DECREF(decoder); |
| if (JSONDecodeError == NULL) |
| return; |
| } |
| exc = PyObject_CallFunction(JSONDecodeError, "(zOn)", msg, s, end); |
| if (exc) { |
| PyErr_SetObject(JSONDecodeError, exc); |
| Py_DECREF(exc); |
| } |
| } |
| |
| static void |
| raise_stop_iteration(Py_ssize_t idx) |
| { |
| PyObject *value = PyLong_FromSsize_t(idx); |
| if (value != NULL) { |
| PyErr_SetObject(PyExc_StopIteration, value); |
| Py_DECREF(value); |
| } |
| } |
| |
| static PyObject * |
| _build_rval_index_tuple(PyObject *rval, Py_ssize_t idx) { |
| /* return (rval, idx) tuple, stealing reference to rval */ |
| PyObject *tpl; |
| PyObject *pyidx; |
| /* |
| steal a reference to rval, returns (rval, idx) |
| */ |
| if (rval == NULL) { |
| return NULL; |
| } |
| pyidx = PyLong_FromSsize_t(idx); |
| if (pyidx == NULL) { |
| Py_DECREF(rval); |
| return NULL; |
| } |
| tpl = PyTuple_New(2); |
| if (tpl == NULL) { |
| Py_DECREF(pyidx); |
| Py_DECREF(rval); |
| return NULL; |
| } |
| PyTuple_SET_ITEM(tpl, 0, rval); |
| PyTuple_SET_ITEM(tpl, 1, pyidx); |
| return tpl; |
| } |
| |
| #define APPEND_OLD_CHUNK \ |
| if (chunk != NULL) { \ |
| if (chunks == NULL) { \ |
| chunks = PyList_New(0); \ |
| if (chunks == NULL) { \ |
| goto bail; \ |
| } \ |
| } \ |
| if (PyList_Append(chunks, chunk)) { \ |
| Py_CLEAR(chunk); \ |
| goto bail; \ |
| } \ |
| Py_CLEAR(chunk); \ |
| } |
| |
| static PyObject * |
| scanstring_unicode(PyObject *pystr, Py_ssize_t end, int strict, Py_ssize_t *next_end_ptr) |
| { |
| /* Read the JSON string from PyUnicode pystr. |
| end is the index of the first character after the quote. |
| if strict is zero then literal control characters are allowed |
| *next_end_ptr is a return-by-reference index of the character |
| after the end quote |
| |
| Return value is a new PyUnicode |
| */ |
| PyObject *rval = NULL; |
| Py_ssize_t len; |
| Py_ssize_t begin = end - 1; |
| Py_ssize_t next /* = begin */; |
| const void *buf; |
| int kind; |
| PyObject *chunks = NULL; |
| PyObject *chunk = NULL; |
| |
| if (PyUnicode_READY(pystr) == -1) |
| return 0; |
| |
| len = PyUnicode_GET_LENGTH(pystr); |
| buf = PyUnicode_DATA(pystr); |
| kind = PyUnicode_KIND(pystr); |
| |
| if (end < 0 || len < end) { |
| PyErr_SetString(PyExc_ValueError, "end is out of bounds"); |
| goto bail; |
| } |
| while (1) { |
| /* Find the end of the string or the next escape */ |
| Py_UCS4 c = 0; |
| for (next = end; next < len; next++) { |
| c = PyUnicode_READ(kind, buf, next); |
| if (c == '"' || c == '\\') { |
| break; |
| } |
| else if (strict && c <= 0x1f) { |
| raise_errmsg("Invalid control character at", pystr, next); |
| goto bail; |
| } |
| } |
| if (!(c == '"' || c == '\\')) { |
| raise_errmsg("Unterminated string starting at", pystr, begin); |
| goto bail; |
| } |
| /* Pick up this chunk if it's not zero length */ |
| if (next != end) { |
| APPEND_OLD_CHUNK |
| chunk = PyUnicode_FromKindAndData( |
| kind, |
| (char*)buf + kind * end, |
| next - end); |
| if (chunk == NULL) { |
| goto bail; |
| } |
| } |
| next++; |
| if (c == '"') { |
| end = next; |
| break; |
| } |
| if (next == len) { |
| raise_errmsg("Unterminated string starting at", pystr, begin); |
| goto bail; |
| } |
| c = PyUnicode_READ(kind, buf, next); |
| if (c != 'u') { |
| /* Non-unicode backslash escapes */ |
| end = next + 1; |
| switch (c) { |
| case '"': break; |
| case '\\': break; |
| case '/': break; |
| case 'b': c = '\b'; break; |
| case 'f': c = '\f'; break; |
| case 'n': c = '\n'; break; |
| case 'r': c = '\r'; break; |
| case 't': c = '\t'; break; |
| default: c = 0; |
| } |
| if (c == 0) { |
| raise_errmsg("Invalid \\escape", pystr, end - 2); |
| goto bail; |
| } |
| } |
| else { |
| c = 0; |
| next++; |
| end = next + 4; |
| if (end >= len) { |
| raise_errmsg("Invalid \\uXXXX escape", pystr, next - 1); |
| goto bail; |
| } |
| /* Decode 4 hex digits */ |
| for (; next < end; next++) { |
| Py_UCS4 digit = PyUnicode_READ(kind, buf, next); |
| c <<= 4; |
| switch (digit) { |
| case '0': case '1': case '2': case '3': case '4': |
| case '5': case '6': case '7': case '8': case '9': |
| c |= (digit - '0'); break; |
| case 'a': case 'b': case 'c': case 'd': case 'e': |
| case 'f': |
| c |= (digit - 'a' + 10); break; |
| case 'A': case 'B': case 'C': case 'D': case 'E': |
| case 'F': |
| c |= (digit - 'A' + 10); break; |
| default: |
| raise_errmsg("Invalid \\uXXXX escape", pystr, end - 5); |
| goto bail; |
| } |
| } |
| /* Surrogate pair */ |
| if (Py_UNICODE_IS_HIGH_SURROGATE(c) && end + 6 < len && |
| PyUnicode_READ(kind, buf, next++) == '\\' && |
| PyUnicode_READ(kind, buf, next++) == 'u') { |
| Py_UCS4 c2 = 0; |
| end += 6; |
| /* Decode 4 hex digits */ |
| for (; next < end; next++) { |
| Py_UCS4 digit = PyUnicode_READ(kind, buf, next); |
| c2 <<= 4; |
| switch (digit) { |
| case '0': case '1': case '2': case '3': case '4': |
| case '5': case '6': case '7': case '8': case '9': |
| c2 |= (digit - '0'); break; |
| case 'a': case 'b': case 'c': case 'd': case 'e': |
| case 'f': |
| c2 |= (digit - 'a' + 10); break; |
| case 'A': case 'B': case 'C': case 'D': case 'E': |
| case 'F': |
| c2 |= (digit - 'A' + 10); break; |
| default: |
| raise_errmsg("Invalid \\uXXXX escape", pystr, end - 5); |
| goto bail; |
| } |
| } |
| if (Py_UNICODE_IS_LOW_SURROGATE(c2)) |
| c = Py_UNICODE_JOIN_SURROGATES(c, c2); |
| else |
| end -= 6; |
| } |
| } |
| APPEND_OLD_CHUNK |
| chunk = PyUnicode_FromKindAndData(PyUnicode_4BYTE_KIND, &c, 1); |
| if (chunk == NULL) { |
| goto bail; |
| } |
| } |
| |
| if (chunks == NULL) { |
| if (chunk != NULL) |
| rval = chunk; |
| else |
| rval = PyUnicode_FromStringAndSize("", 0); |
| } |
| else { |
| APPEND_OLD_CHUNK |
| rval = join_list_unicode(chunks); |
| if (rval == NULL) { |
| goto bail; |
| } |
| Py_CLEAR(chunks); |
| } |
| |
| *next_end_ptr = end; |
| return rval; |
| bail: |
| *next_end_ptr = -1; |
| Py_XDECREF(chunks); |
| Py_XDECREF(chunk); |
| return NULL; |
| } |
| |
| PyDoc_STRVAR(pydoc_scanstring, |
| "scanstring(string, end, strict=True) -> (string, end)\n" |
| "\n" |
| "Scan the string s for a JSON string. End is the index of the\n" |
| "character in s after the quote that started the JSON string.\n" |
| "Unescapes all valid JSON string escape sequences and raises ValueError\n" |
| "on attempt to decode an invalid string. If strict is False then literal\n" |
| "control characters are allowed in the string.\n" |
| "\n" |
| "Returns a tuple of the decoded string and the index of the character in s\n" |
| "after the end quote." |
| ); |
| |
| static PyObject * |
| py_scanstring(PyObject* self UNUSED, PyObject *args) |
| { |
| PyObject *pystr; |
| PyObject *rval; |
| Py_ssize_t end; |
| Py_ssize_t next_end = -1; |
| int strict = 1; |
| if (!PyArg_ParseTuple(args, "On|i:scanstring", &pystr, &end, &strict)) { |
| return NULL; |
| } |
| if (PyUnicode_Check(pystr)) { |
| rval = scanstring_unicode(pystr, end, strict, &next_end); |
| } |
| else { |
| PyErr_Format(PyExc_TypeError, |
| "first argument must be a string, not %.80s", |
| Py_TYPE(pystr)->tp_name); |
| return NULL; |
| } |
| return _build_rval_index_tuple(rval, next_end); |
| } |
| |
| PyDoc_STRVAR(pydoc_encode_basestring_ascii, |
| "encode_basestring_ascii(string) -> string\n" |
| "\n" |
| "Return an ASCII-only JSON representation of a Python string" |
| ); |
| |
| static PyObject * |
| py_encode_basestring_ascii(PyObject* self UNUSED, PyObject *pystr) |
| { |
| PyObject *rval; |
| /* Return an ASCII-only JSON representation of a Python string */ |
| /* METH_O */ |
| if (PyUnicode_Check(pystr)) { |
| rval = ascii_escape_unicode(pystr); |
| } |
| else { |
| PyErr_Format(PyExc_TypeError, |
| "first argument must be a string, not %.80s", |
| Py_TYPE(pystr)->tp_name); |
| return NULL; |
| } |
| return rval; |
| } |
| |
| |
| PyDoc_STRVAR(pydoc_encode_basestring, |
| "encode_basestring(string) -> string\n" |
| "\n" |
| "Return a JSON representation of a Python string" |
| ); |
| |
| static PyObject * |
| py_encode_basestring(PyObject* self UNUSED, PyObject *pystr) |
| { |
| PyObject *rval; |
| /* Return a JSON representation of a Python string */ |
| /* METH_O */ |
| if (PyUnicode_Check(pystr)) { |
| rval = escape_unicode(pystr); |
| } |
| else { |
| PyErr_Format(PyExc_TypeError, |
| "first argument must be a string, not %.80s", |
| Py_TYPE(pystr)->tp_name); |
| return NULL; |
| } |
| return rval; |
| } |
| |
| static void |
| scanner_dealloc(PyObject *self) |
| { |
| /* Deallocate scanner object */ |
| scanner_clear(self); |
| Py_TYPE(self)->tp_free(self); |
| } |
| |
| static int |
| scanner_traverse(PyObject *self, visitproc visit, void *arg) |
| { |
| PyScannerObject *s; |
| assert(PyScanner_Check(self)); |
| s = (PyScannerObject *)self; |
| Py_VISIT(s->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; |
| } |
| |
| static int |
| scanner_clear(PyObject *self) |
| { |
| 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); |
| Py_CLEAR(s->memo); |
| return 0; |
| } |
| |
| static PyObject * |
| _parse_object_unicode(PyScannerObject *s, PyObject *pystr, Py_ssize_t idx, Py_ssize_t *next_idx_ptr) { |
| /* Read a JSON object from PyUnicode pystr. |
| idx is the index of the first character after the opening curly brace. |
| *next_idx_ptr is a return-by-reference index to the first character after |
| the closing curly brace. |
| |
| Returns a new PyObject (usually a dict, but object_hook can change that) |
| */ |
| void *str; |
| int kind; |
| Py_ssize_t end_idx; |
| PyObject *val = NULL; |
| PyObject *rval = NULL; |
| PyObject *key = NULL; |
| int strict = PyObject_IsTrue(s->strict); |
| int has_pairs_hook = (s->object_pairs_hook != Py_None); |
| Py_ssize_t next_idx; |
| |
| if (strict < 0) |
| return NULL; |
| |
| if (PyUnicode_READY(pystr) == -1) |
| return NULL; |
| |
| str = PyUnicode_DATA(pystr); |
| kind = PyUnicode_KIND(pystr); |
| end_idx = PyUnicode_GET_LENGTH(pystr) - 1; |
| |
| if (has_pairs_hook) |
| rval = PyList_New(0); |
| else |
| rval = PyDict_New(); |
| if (rval == NULL) |
| return NULL; |
| |
| /* skip whitespace after { */ |
| while (idx <= end_idx && IS_WHITESPACE(PyUnicode_READ(kind,str, idx))) idx++; |
| |
| /* only loop if the object is non-empty */ |
| if (idx > end_idx || PyUnicode_READ(kind, str, idx) != '}') { |
| while (1) { |
| PyObject *memokey; |
| |
| /* read key */ |
| if (idx > end_idx || PyUnicode_READ(kind, str, idx) != '"') { |
| raise_errmsg("Expecting property name enclosed in double quotes", pystr, idx); |
| goto bail; |
| } |
| key = scanstring_unicode(pystr, idx + 1, strict, &next_idx); |
| if (key == NULL) |
| goto bail; |
| memokey = PyDict_GetItem(s->memo, key); |
| if (memokey != NULL) { |
| Py_INCREF(memokey); |
| Py_DECREF(key); |
| key = memokey; |
| } |
| else { |
| if (PyDict_SetItem(s->memo, key, key) < 0) |
| goto bail; |
| } |
| idx = next_idx; |
| |
| /* skip whitespace between key and : delimiter, read :, skip whitespace */ |
| while (idx <= end_idx && IS_WHITESPACE(PyUnicode_READ(kind, str, idx))) idx++; |
| if (idx > end_idx || PyUnicode_READ(kind, str, idx) != ':') { |
| raise_errmsg("Expecting ':' delimiter", pystr, idx); |
| goto bail; |
| } |
| idx++; |
| while (idx <= end_idx && IS_WHITESPACE(PyUnicode_READ(kind, str, idx))) idx++; |
| |
| /* read any JSON term */ |
| val = scan_once_unicode(s, pystr, idx, &next_idx); |
| if (val == NULL) |
| goto bail; |
| |
| if (has_pairs_hook) { |
| PyObject *item = PyTuple_Pack(2, key, val); |
| if (item == NULL) |
| goto bail; |
| Py_CLEAR(key); |
| Py_CLEAR(val); |
| if (PyList_Append(rval, item) == -1) { |
| Py_DECREF(item); |
| goto bail; |
| } |
| Py_DECREF(item); |
| } |
| else { |
| if (PyDict_SetItem(rval, key, val) < 0) |
| goto bail; |
| Py_CLEAR(key); |
| Py_CLEAR(val); |
| } |
| idx = next_idx; |
| |
| /* skip whitespace before } or , */ |
| while (idx <= end_idx && IS_WHITESPACE(PyUnicode_READ(kind, str, idx))) idx++; |
| |
| /* bail if the object is closed or we didn't get the , delimiter */ |
| if (idx <= end_idx && PyUnicode_READ(kind, str, idx) == '}') |
| break; |
| if (idx > end_idx || PyUnicode_READ(kind, str, idx) != ',') { |
| raise_errmsg("Expecting ',' delimiter", pystr, idx); |
| goto bail; |
| } |
| idx++; |
| |
| /* skip whitespace after , delimiter */ |
| while (idx <= end_idx && IS_WHITESPACE(PyUnicode_READ(kind, str, idx))) idx++; |
| } |
| } |
| |
| *next_idx_ptr = idx + 1; |
| |
| if (has_pairs_hook) { |
| val = PyObject_CallFunctionObjArgs(s->object_pairs_hook, rval, NULL); |
| Py_DECREF(rval); |
| return val; |
| } |
| |
| /* if object_hook is not None: rval = object_hook(rval) */ |
| if (s->object_hook != Py_None) { |
| val = PyObject_CallFunctionObjArgs(s->object_hook, rval, NULL); |
| Py_DECREF(rval); |
| return val; |
| } |
| return rval; |
| bail: |
| Py_XDECREF(key); |
| Py_XDECREF(val); |
| Py_XDECREF(rval); |
| return NULL; |
| } |
| |
| static PyObject * |
| _parse_array_unicode(PyScannerObject *s, PyObject *pystr, Py_ssize_t idx, Py_ssize_t *next_idx_ptr) { |
| /* Read a JSON array from PyUnicode pystr. |
| idx is the index of the first character after the opening brace. |
| *next_idx_ptr is a return-by-reference index to the first character after |
| the closing brace. |
| |
| Returns a new PyList |
| */ |
| void *str; |
| int kind; |
| Py_ssize_t end_idx; |
| PyObject *val = NULL; |
| PyObject *rval = PyList_New(0); |
| Py_ssize_t next_idx; |
| if (rval == NULL) |
| return NULL; |
| |
| if (PyUnicode_READY(pystr) == -1) |
| return NULL; |
| |
| str = PyUnicode_DATA(pystr); |
| kind = PyUnicode_KIND(pystr); |
| end_idx = PyUnicode_GET_LENGTH(pystr) - 1; |
| |
| /* skip whitespace after [ */ |
| while (idx <= end_idx && IS_WHITESPACE(PyUnicode_READ(kind, str, idx))) idx++; |
| |
| /* only loop if the array is non-empty */ |
| if (idx > end_idx || PyUnicode_READ(kind, str, idx) != ']') { |
| while (1) { |
| |
| /* read any JSON term */ |
| val = scan_once_unicode(s, pystr, idx, &next_idx); |
| if (val == NULL) |
| goto bail; |
| |
| if (PyList_Append(rval, val) == -1) |
| goto bail; |
| |
| Py_CLEAR(val); |
| idx = next_idx; |
| |
| /* skip whitespace between term and , */ |
| while (idx <= end_idx && IS_WHITESPACE(PyUnicode_READ(kind, str, idx))) idx++; |
| |
| /* bail if the array is closed or we didn't get the , delimiter */ |
| if (idx <= end_idx && PyUnicode_READ(kind, str, idx) == ']') |
| break; |
| if (idx > end_idx || PyUnicode_READ(kind, str, idx) != ',') { |
| raise_errmsg("Expecting ',' delimiter", pystr, idx); |
| goto bail; |
| } |
| idx++; |
| |
| /* skip whitespace after , */ |
| while (idx <= end_idx && IS_WHITESPACE(PyUnicode_READ(kind, str, idx))) idx++; |
| } |
| } |
| |
| /* verify that idx < end_idx, PyUnicode_READ(kind, str, idx) should be ']' */ |
| if (idx > end_idx || PyUnicode_READ(kind, str, idx) != ']') { |
| raise_errmsg("Expecting value", pystr, end_idx); |
| goto bail; |
| } |
| *next_idx_ptr = idx + 1; |
| return rval; |
| bail: |
| Py_XDECREF(val); |
| Py_DECREF(rval); |
| return NULL; |
| } |
| |
| static PyObject * |
| _parse_constant(PyScannerObject *s, const char *constant, Py_ssize_t idx, Py_ssize_t *next_idx_ptr) { |
| /* Read a JSON constant. |
| constant is the constant string that was found |
| ("NaN", "Infinity", "-Infinity"). |
| idx is the index of the first character of the constant |
| *next_idx_ptr is a return-by-reference index to the first character after |
| the constant. |
| |
| Returns the result of parse_constant |
| */ |
| PyObject *cstr; |
| PyObject *rval; |
| /* constant is "NaN", "Infinity", or "-Infinity" */ |
| cstr = PyUnicode_InternFromString(constant); |
| if (cstr == NULL) |
| return NULL; |
| |
| /* rval = parse_constant(constant) */ |
| rval = PyObject_CallFunctionObjArgs(s->parse_constant, cstr, NULL); |
| idx += PyUnicode_GET_LENGTH(cstr); |
| Py_DECREF(cstr); |
| *next_idx_ptr = idx; |
| return rval; |
| } |
| |
| static PyObject * |
| _match_number_unicode(PyScannerObject *s, PyObject *pystr, Py_ssize_t start, Py_ssize_t *next_idx_ptr) { |
| /* Read a JSON number from PyUnicode pystr. |
| idx is the index of the first character of the number |
| *next_idx_ptr is a return-by-reference index to the first character after |
| the number. |
| |
| Returns a new PyObject representation of that number: |
| PyLong, or PyFloat. |
| May return other types if parse_int or parse_float are set |
| */ |
| void *str; |
| int kind; |
| Py_ssize_t end_idx; |
| Py_ssize_t idx = start; |
| int is_float = 0; |
| PyObject *rval; |
| PyObject *numstr = NULL; |
| PyObject *custom_func; |
| |
| if (PyUnicode_READY(pystr) == -1) |
| return NULL; |
| |
| str = PyUnicode_DATA(pystr); |
| kind = PyUnicode_KIND(pystr); |
| end_idx = PyUnicode_GET_LENGTH(pystr) - 1; |
| |
| /* read a sign if it's there, make sure it's not the end of the string */ |
| if (PyUnicode_READ(kind, str, idx) == '-') { |
| idx++; |
| if (idx > end_idx) { |
| raise_stop_iteration(start); |
| return NULL; |
| } |
| } |
| |
| /* read as many integer digits as we find as long as it doesn't start with 0 */ |
| if (PyUnicode_READ(kind, str, idx) >= '1' && PyUnicode_READ(kind, str, idx) <= '9') { |
| idx++; |
| while (idx <= end_idx && PyUnicode_READ(kind, str, idx) >= '0' && PyUnicode_READ(kind, str, idx) <= '9') idx++; |
| } |
| /* if it starts with 0 we only expect one integer digit */ |
| else if (PyUnicode_READ(kind, str, idx) == '0') { |
| idx++; |
| } |
| /* no integer digits, error */ |
| else { |
| raise_stop_iteration(start); |
| return NULL; |
| } |
| |
| /* if the next char is '.' followed by a digit then read all float digits */ |
| if (idx < end_idx && PyUnicode_READ(kind, str, idx) == '.' && PyUnicode_READ(kind, str, idx + 1) >= '0' && PyUnicode_READ(kind, str, idx + 1) <= '9') { |
| is_float = 1; |
| idx += 2; |
| while (idx <= end_idx && PyUnicode_READ(kind, str, idx) >= '0' && PyUnicode_READ(kind, str, idx) <= '9') idx++; |
| } |
| |
| /* if the next char is 'e' or 'E' then maybe read the exponent (or backtrack) */ |
| if (idx < end_idx && (PyUnicode_READ(kind, str, idx) == 'e' || PyUnicode_READ(kind, str, idx) == 'E')) { |
| Py_ssize_t e_start = idx; |
| idx++; |
| |
| /* read an exponent sign if present */ |
| if (idx < end_idx && (PyUnicode_READ(kind, str, idx) == '-' || PyUnicode_READ(kind, str, idx) == '+')) idx++; |
| |
| /* read all digits */ |
| while (idx <= end_idx && PyUnicode_READ(kind, str, idx) >= '0' && PyUnicode_READ(kind, str, idx) <= '9') idx++; |
| |
| /* if we got a digit, then parse as float. if not, backtrack */ |
| if (PyUnicode_READ(kind, str, idx - 1) >= '0' && PyUnicode_READ(kind, str, idx - 1) <= '9') { |
| is_float = 1; |
| } |
| else { |
| idx = e_start; |
| } |
| } |
| |
| if (is_float && s->parse_float != (PyObject *)&PyFloat_Type) |
| custom_func = s->parse_float; |
| else if (!is_float && s->parse_int != (PyObject *) &PyLong_Type) |
| custom_func = s->parse_int; |
| else |
| custom_func = NULL; |
| |
| if (custom_func) { |
| /* copy the section we determined to be a number */ |
| numstr = PyUnicode_FromKindAndData(kind, |
| (char*)str + kind * start, |
| idx - start); |
| if (numstr == NULL) |
| return NULL; |
| rval = PyObject_CallFunctionObjArgs(custom_func, numstr, NULL); |
| } |
| else { |
| Py_ssize_t i, n; |
| char *buf; |
| /* Straight conversion to ASCII, to avoid costly conversion of |
| decimal unicode digits (which cannot appear here) */ |
| n = idx - start; |
| numstr = PyBytes_FromStringAndSize(NULL, n); |
| if (numstr == NULL) |
| return NULL; |
| buf = PyBytes_AS_STRING(numstr); |
| for (i = 0; i < n; i++) { |
| buf[i] = (char) PyUnicode_READ(kind, str, i + start); |
| } |
| if (is_float) |
| rval = PyFloat_FromString(numstr); |
| else |
| rval = PyLong_FromString(buf, NULL, 10); |
| } |
| Py_DECREF(numstr); |
| *next_idx_ptr = idx; |
| return rval; |
| } |
| |
| static PyObject * |
| scan_once_unicode(PyScannerObject *s, PyObject *pystr, Py_ssize_t idx, Py_ssize_t *next_idx_ptr) |
| { |
| /* Read one JSON term (of any kind) from PyUnicode pystr. |
| idx is the index of the first character of the term |
| *next_idx_ptr is a return-by-reference index to the first character after |
| the number. |
| |
| Returns a new PyObject representation of the term. |
| */ |
| PyObject *res; |
| void *str; |
| int kind; |
| Py_ssize_t length; |
| int strict; |
| |
| if (PyUnicode_READY(pystr) == -1) |
| return NULL; |
| |
| str = PyUnicode_DATA(pystr); |
| kind = PyUnicode_KIND(pystr); |
| length = PyUnicode_GET_LENGTH(pystr); |
| |
| if (idx < 0) { |
| PyErr_SetString(PyExc_ValueError, "idx cannot be negative"); |
| return NULL; |
| } |
| if (idx >= length) { |
| raise_stop_iteration(idx); |
| return NULL; |
| } |
| |
| switch (PyUnicode_READ(kind, str, idx)) { |
| case '"': |
| /* string */ |
| strict = PyObject_IsTrue(s->strict); |
| if (strict < 0) |
| return NULL; |
| return scanstring_unicode(pystr, idx + 1, strict, next_idx_ptr); |
| case '{': |
| /* object */ |
| if (Py_EnterRecursiveCall(" while decoding a JSON object " |
| "from a unicode string")) |
| return NULL; |
| res = _parse_object_unicode(s, pystr, idx + 1, next_idx_ptr); |
| Py_LeaveRecursiveCall(); |
| return res; |
| case '[': |
| /* array */ |
| if (Py_EnterRecursiveCall(" while decoding a JSON array " |
| "from a unicode string")) |
| return NULL; |
| res = _parse_array_unicode(s, pystr, idx + 1, next_idx_ptr); |
| Py_LeaveRecursiveCall(); |
| return res; |
| case 'n': |
| /* null */ |
| if ((idx + 3 < length) && PyUnicode_READ(kind, str, idx + 1) == 'u' && PyUnicode_READ(kind, str, idx + 2) == 'l' && PyUnicode_READ(kind, str, idx + 3) == 'l') { |
| Py_INCREF(Py_None); |
| *next_idx_ptr = idx + 4; |
| return Py_None; |
| } |
| break; |
| case 't': |
| /* true */ |
| if ((idx + 3 < length) && PyUnicode_READ(kind, str, idx + 1) == 'r' && PyUnicode_READ(kind, str, idx + 2) == 'u' && PyUnicode_READ(kind, str, idx + 3) == 'e') { |
| Py_INCREF(Py_True); |
| *next_idx_ptr = idx + 4; |
| return Py_True; |
| } |
| break; |
| case 'f': |
| /* false */ |
| if ((idx + 4 < length) && PyUnicode_READ(kind, str, idx + 1) == 'a' && |
| PyUnicode_READ(kind, str, idx + 2) == 'l' && |
| PyUnicode_READ(kind, str, idx + 3) == 's' && |
| PyUnicode_READ(kind, str, idx + 4) == 'e') { |
| Py_INCREF(Py_False); |
| *next_idx_ptr = idx + 5; |
| return Py_False; |
| } |
| break; |
| case 'N': |
| /* NaN */ |
| if ((idx + 2 < length) && PyUnicode_READ(kind, str, idx + 1) == 'a' && |
| PyUnicode_READ(kind, str, idx + 2) == 'N') { |
| return _parse_constant(s, "NaN", idx, next_idx_ptr); |
| } |
| break; |
| case 'I': |
| /* Infinity */ |
| if ((idx + 7 < length) && PyUnicode_READ(kind, str, idx + 1) == 'n' && |
| PyUnicode_READ(kind, str, idx + 2) == 'f' && |
| PyUnicode_READ(kind, str, idx + 3) == 'i' && |
| PyUnicode_READ(kind, str, idx + 4) == 'n' && |
| PyUnicode_READ(kind, str, idx + 5) == 'i' && |
| PyUnicode_READ(kind, str, idx + 6) == 't' && |
| PyUnicode_READ(kind, str, idx + 7) == 'y') { |
| return _parse_constant(s, "Infinity", idx, next_idx_ptr); |
| } |
| break; |
| case '-': |
| /* -Infinity */ |
| if ((idx + 8 < length) && PyUnicode_READ(kind, str, idx + 1) == 'I' && |
| PyUnicode_READ(kind, str, idx + 2) == 'n' && |
| PyUnicode_READ(kind, str, idx + 3) == 'f' && |
| PyUnicode_READ(kind, str, idx + 4) == 'i' && |
| PyUnicode_READ(kind, str, idx + 5) == 'n' && |
| PyUnicode_READ(kind, str, idx + 6) == 'i' && |
| PyUnicode_READ(kind, str, idx + 7) == 't' && |
| PyUnicode_READ(kind, str, idx + 8) == 'y') { |
| return _parse_constant(s, "-Infinity", idx, next_idx_ptr); |
| } |
| break; |
| } |
| /* Didn't find a string, object, array, or named constant. Look for a number. */ |
| return _match_number_unicode(s, pystr, idx, next_idx_ptr); |
| } |
| |
| static PyObject * |
| scanner_call(PyObject *self, PyObject *args, PyObject *kwds) |
| { |
| /* Python callable interface to scan_once_{str,unicode} */ |
| PyObject *pystr; |
| PyObject *rval; |
| Py_ssize_t idx; |
| Py_ssize_t next_idx = -1; |
| static char *kwlist[] = {"string", "idx", NULL}; |
| PyScannerObject *s; |
| assert(PyScanner_Check(self)); |
| s = (PyScannerObject *)self; |
| if (!PyArg_ParseTupleAndKeywords(args, kwds, "On:scan_once", kwlist, &pystr, &idx)) |
| return NULL; |
| |
| if (PyUnicode_Check(pystr)) { |
| rval = scan_once_unicode(s, pystr, idx, &next_idx); |
| } |
| else { |
| PyErr_Format(PyExc_TypeError, |
| "first argument must be a string, not %.80s", |
| Py_TYPE(pystr)->tp_name); |
| return NULL; |
| } |
| PyDict_Clear(s->memo); |
| if (rval == NULL) |
| return NULL; |
| return _build_rval_index_tuple(rval, next_idx); |
| } |
| |
| static PyObject * |
| scanner_new(PyTypeObject *type, PyObject *args, PyObject *kwds) |
| { |
| PyScannerObject *s; |
| 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; |
| |
| if (s->memo == NULL) { |
| s->memo = PyDict_New(); |
| if (s->memo == NULL) |
| goto bail; |
| } |
| |
| /* All of these will fail "gracefully" so we don't need to verify them */ |
| 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; |
| PyObject *markers, *defaultfn, *encoder, *indent, *key_separator; |
| PyObject *item_separator, *sort_keys, *skipkeys; |
| int allow_nan; |
| |
| assert(PyEncoder_Check(self)); |
| s = (PyEncoderObject *)self; |
| |
| if (!PyArg_ParseTupleAndKeywords(args, kwds, "OOOOUUOOp:make_encoder", kwlist, |
| &markers, &defaultfn, &encoder, &indent, |
| &key_separator, &item_separator, |
| &sort_keys, &skipkeys, &allow_nan)) |
| return -1; |
| |
| if (markers != Py_None && !PyDict_Check(markers)) { |
| PyErr_Format(PyExc_TypeError, |
| "make_encoder() argument 1 must be dict or None, " |
| "not %.200s", Py_TYPE(markers)->tp_name); |
| return -1; |
| } |
| |
| 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 = NULL; |
| if (PyCFunction_Check(s->encoder)) { |
| PyCFunction f = PyCFunction_GetFunction(s->encoder); |
| if (f == (PyCFunction)py_encode_basestring_ascii || |
| f == (PyCFunction)py_encode_basestring) { |
| s->fast_encode = f; |
| } |
| } |
| s->allow_nan = allow_nan; |
| |
| 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; |
| Py_ssize_t indent_level; |
| PyEncoderObject *s; |
| _PyAccu acc; |
| |
| assert(PyEncoder_Check(self)); |
| s = (PyEncoderObject *)self; |
| if (!PyArg_ParseTupleAndKeywords(args, kwds, "On:_iterencode", kwlist, |
| &obj, &indent_level)) |
| return NULL; |
| if (_PyAccu_Init(&acc)) |
| return NULL; |
| if (encoder_listencode_obj(s, &acc, obj, indent_level)) { |
| _PyAccu_Destroy(&acc); |
| return NULL; |
| } |
| return _PyAccu_FinishAsList(&acc); |
| } |
| |
| static PyObject * |
| _encoded_const(PyObject *obj) |
| { |
| /* Return the JSON string representation of None, True, False */ |
| if (obj == Py_None) { |
| static PyObject *s_null = NULL; |
| if (s_null == NULL) { |
| s_null = PyUnicode_InternFromString("null"); |
| } |
| Py_INCREF(s_null); |
| return s_null; |
| } |
| else if (obj == Py_True) { |
| static PyObject *s_true = NULL; |
| if (s_true == NULL) { |
| s_true = PyUnicode_InternFromString("true"); |
| } |
| Py_INCREF(s_true); |
| return s_true; |
| } |
| else if (obj == Py_False) { |
| static PyObject *s_false = NULL; |
| if (s_false == NULL) { |
| s_false = PyUnicode_InternFromString("false"); |
| } |
| Py_INCREF(s_false); |
| return s_false; |
| } |
| else { |
| PyErr_SetString(PyExc_ValueError, "not a const"); |
| return NULL; |
| } |
| } |
| |
| static PyObject * |
| encoder_encode_long(PyEncoderObject* s UNUSED, PyObject *obj) |
| { |
| /* Return the JSON representation of a PyLong and PyLong subclasses. |
| Calls int() on PyLong subclasses in case the str() was changed. |
| Added specifically to deal with IntEnum. See Issue18264. */ |
| PyObject *encoded, *longobj; |
| if (PyLong_CheckExact(obj)) { |
| encoded = PyObject_Str(obj); |
| } |
| else { |
| longobj = PyNumber_Long(obj); |
| if (longobj == NULL) { |
| PyErr_SetString( |
| PyExc_ValueError, |
| "Unable to coerce int subclass to int" |
| ); |
| return NULL; |
| } |
| encoded = PyObject_Str(longobj); |
| Py_DECREF(longobj); |
| } |
| return encoded; |
| } |
| |
| |
| static PyObject * |
| encoder_encode_float(PyEncoderObject *s, PyObject *obj) |
| { |
| /* Return the JSON representation of a PyFloat. |
| Modified to call float() on float subclasses in case the subclass |
| changes the repr. See Issue18264. */ |
| PyObject *encoded, *floatobj; |
| 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"); |
| } |
| } |
| /* coerce float subclasses to float (primarily for Enum) */ |
| if (PyFloat_CheckExact(obj)) { |
| /* Use a better float format here? */ |
| encoded = PyObject_Repr(obj); |
| } |
| else { |
| floatobj = PyNumber_Float(obj); |
| if (floatobj == NULL) { |
| PyErr_SetString( |
| PyExc_ValueError, |
| "Unable to coerce float subclass to float" |
| ); |
| return NULL; |
| } |
| encoded = PyObject_Repr(floatobj); |
| Py_DECREF(floatobj); |
| } |
| return encoded; |
| } |
| |
| static PyObject * |
| encoder_encode_string(PyEncoderObject *s, PyObject *obj) |
| { |
| /* Return the JSON representation of a string */ |
| if (s->fast_encode) |
| return s->fast_encode(NULL, obj); |
| else |
| return PyObject_CallFunctionObjArgs(s->encoder, obj, NULL); |
| } |
| |
| static int |
| _steal_accumulate(_PyAccu *acc, PyObject *stolen) |
| { |
| /* Append stolen and then decrement its reference count */ |
| int rval = _PyAccu_Accumulate(acc, stolen); |
| Py_DECREF(stolen); |
| return rval; |
| } |
| |
| static int |
| encoder_listencode_obj(PyEncoderObject *s, _PyAccu *acc, |
| PyObject *obj, Py_ssize_t indent_level) |
| { |
| /* Encode Python object obj to a JSON term */ |
| PyObject *newobj; |
| int rv; |
| |
| if (obj == Py_None || obj == Py_True || obj == Py_False) { |
| PyObject *cstr = _encoded_const(obj); |
| if (cstr == NULL) |
| return -1; |
| return _steal_accumulate(acc, cstr); |
| } |
| else if (PyUnicode_Check(obj)) |
| { |
| PyObject *encoded = encoder_encode_string(s, obj); |
| if (encoded == NULL) |
| return -1; |
| return _steal_accumulate(acc, encoded); |
| } |
| else if (PyLong_Check(obj)) { |
| PyObject *encoded = encoder_encode_long(s, obj); |
| if (encoded == NULL) |
| return -1; |
| return _steal_accumulate(acc, encoded); |
| } |
| else if (PyFloat_Check(obj)) { |
| PyObject *encoded = encoder_encode_float(s, obj); |
| if (encoded == NULL) |
| return -1; |
| return _steal_accumulate(acc, encoded); |
| } |
| else if (PyList_Check(obj) || PyTuple_Check(obj)) { |
| if (Py_EnterRecursiveCall(" while encoding a JSON object")) |
| return -1; |
| rv = encoder_listencode_list(s, acc, obj, indent_level); |
| Py_LeaveRecursiveCall(); |
| return rv; |
| } |
| else if (PyDict_Check(obj)) { |
| if (Py_EnterRecursiveCall(" while encoding a JSON object")) |
| return -1; |
| rv = encoder_listencode_dict(s, acc, obj, indent_level); |
| Py_LeaveRecursiveCall(); |
| return rv; |
| } |
| else { |
| PyObject *ident = NULL; |
| if (s->markers != Py_None) { |
| int has_key; |
| ident = PyLong_FromVoidPtr(obj); |
| if (ident == NULL) |
| return -1; |
| has_key = PyDict_Contains(s->markers, ident); |
| if (has_key) { |
| if (has_key != -1) |
| PyErr_SetString(PyExc_ValueError, "Circular reference detected"); |
| Py_DECREF(ident); |
| return -1; |
| } |
| if (PyDict_SetItem(s->markers, ident, obj)) { |
| Py_DECREF(ident); |
| return -1; |
| } |
| } |
| newobj = PyObject_CallFunctionObjArgs(s->defaultfn, obj, NULL); |
| if (newobj == NULL) { |
| Py_XDECREF(ident); |
| return -1; |
| } |
| |
| if (Py_EnterRecursiveCall(" while encoding a JSON object")) |
| return -1; |
| rv = encoder_listencode_obj(s, acc, newobj, indent_level); |
| Py_LeaveRecursiveCall(); |
| |
| Py_DECREF(newobj); |
| if (rv) { |
| Py_XDECREF(ident); |
| return -1; |
| } |
| if (ident != NULL) { |
| if (PyDict_DelItem(s->markers, ident)) { |
| Py_XDECREF(ident); |
| return -1; |
| } |
| Py_XDECREF(ident); |
| } |
| return rv; |
| } |
| } |
| |
| static int |
| encoder_listencode_dict(PyEncoderObject *s, _PyAccu *acc, |
| PyObject *dct, Py_ssize_t indent_level) |
| { |
| /* Encode Python dict dct a JSON term */ |
| static PyObject *open_dict = NULL; |
| static PyObject *close_dict = NULL; |
| static PyObject *empty_dict = NULL; |
| PyObject *kstr = NULL; |
| PyObject *ident = NULL; |
| PyObject *it = NULL; |
| PyObject *items; |
| PyObject *item = NULL; |
| int skipkeys; |
| int sortkeys; |
| Py_ssize_t idx; |
| |
| if (open_dict == NULL || close_dict == NULL || empty_dict == NULL) { |
| open_dict = PyUnicode_InternFromString("{"); |
| close_dict = PyUnicode_InternFromString("}"); |
| empty_dict = PyUnicode_InternFromString("{}"); |
| if (open_dict == NULL || close_dict == NULL || empty_dict == NULL) |
| return -1; |
| } |
| if (Py_SIZE(dct) == 0) |
| return _PyAccu_Accumulate(acc, empty_dict); |
| |
| if (s->markers != Py_None) { |
| int has_key; |
| ident = PyLong_FromVoidPtr(dct); |
| if (ident == NULL) |
| goto bail; |
| has_key = PyDict_Contains(s->markers, ident); |
| if (has_key) { |
| if (has_key != -1) |
| PyErr_SetString(PyExc_ValueError, "Circular reference detected"); |
| goto bail; |
| } |
| if (PyDict_SetItem(s->markers, ident, dct)) { |
| goto bail; |
| } |
| } |
| |
| if (_PyAccu_Accumulate(acc, open_dict)) |
| goto bail; |
| |
| if (s->indent != Py_None) { |
| /* TODO: DOES NOT RUN */ |
| indent_level += 1; |
| /* |
| newline_indent = '\n' + (' ' * (_indent * _current_indent_level)) |
| separator = _item_separator + newline_indent |
| buf += newline_indent |
| */ |
| } |
| |
| items = PyMapping_Items(dct); |
| if (items == NULL) |
| goto bail; |
| sortkeys = PyObject_IsTrue(s->sort_keys); |
| if (sortkeys < 0 || (sortkeys && PyList_Sort(items) < 0)) |
| goto bail; |
| it = PyObject_GetIter(items); |
| Py_DECREF(items); |
| if (it == NULL) |
| goto bail; |
| skipkeys = PyObject_IsTrue(s->skipkeys); |
| if (skipkeys < 0) |
| goto bail; |
| idx = 0; |
| 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); |
| if (PyUnicode_Check(key)) { |
| Py_INCREF(key); |
| kstr = key; |
| } |
| else if (PyFloat_Check(key)) { |
| kstr = encoder_encode_float(s, key); |
| if (kstr == NULL) |
| goto bail; |
| } |
| else if (key == Py_True || key == Py_False || key == Py_None) { |
| /* This must come before the PyLong_Check because |
| True and False are also 1 and 0.*/ |
| kstr = _encoded_const(key); |
| if (kstr == NULL) |
| goto bail; |
| } |
| else if (PyLong_Check(key)) { |
| kstr = encoder_encode_long(s, key); |
| if (kstr == NULL) { |
| goto bail; |
| } |
| } |
| else if (skipkeys) { |
| Py_DECREF(item); |
| continue; |
| } |
| else { |
| /* TODO: include repr of key */ |
| PyErr_SetString(PyExc_TypeError, "keys must be a string"); |
| goto bail; |
| } |
| |
| if (idx) { |
| if (_PyAccu_Accumulate(acc, s->item_separator)) |
| goto bail; |
| } |
| |
| encoded = encoder_encode_string(s, kstr); |
| Py_CLEAR(kstr); |
| if (encoded == NULL) |
| goto bail; |
| if (_PyAccu_Accumulate(acc, encoded)) { |
| Py_DECREF(encoded); |
| goto bail; |
| } |
| Py_DECREF(encoded); |
| if (_PyAccu_Accumulate(acc, s->key_separator)) |
| goto bail; |
| |
| value = PyTuple_GET_ITEM(item, 1); |
| if (encoder_listencode_obj(s, acc, value, indent_level)) |
| goto bail; |
| idx += 1; |
| Py_DECREF(item); |
| } |
| if (PyErr_Occurred()) |
| goto bail; |
| Py_CLEAR(it); |
| |
| if (ident != NULL) { |
| if (PyDict_DelItem(s->markers, ident)) |
| goto bail; |
| Py_CLEAR(ident); |
| } |
| /* TODO DOES NOT RUN; dead code |
| if (s->indent != Py_None) { |
| indent_level -= 1; |
| |
| yield '\n' + (' ' * (_indent * _current_indent_level)) |
| }*/ |
| if (_PyAccu_Accumulate(acc, close_dict)) |
| goto bail; |
| return 0; |
| |
| bail: |
| Py_XDECREF(it); |
| Py_XDECREF(item); |
| Py_XDECREF(kstr); |
| Py_XDECREF(ident); |
| return -1; |
| } |
| |
| |
| static int |
| encoder_listencode_list(PyEncoderObject *s, _PyAccu *acc, |
| PyObject *seq, Py_ssize_t indent_level) |
| { |
| /* Encode Python list seq to a JSON term */ |
| static PyObject *open_array = NULL; |
| static PyObject *close_array = NULL; |
| static PyObject *empty_array = NULL; |
| PyObject *ident = NULL; |
| PyObject *s_fast = NULL; |
| Py_ssize_t i; |
| |
| if (open_array == NULL || close_array == NULL || empty_array == NULL) { |
| open_array = PyUnicode_InternFromString("["); |
| close_array = PyUnicode_InternFromString("]"); |
| empty_array = PyUnicode_InternFromString("[]"); |
| if (open_array == NULL || close_array == NULL || empty_array == NULL) |
| return -1; |
| } |
| ident = NULL; |
| s_fast = PySequence_Fast(seq, "_iterencode_list needs a sequence"); |
| if (s_fast == NULL) |
| return -1; |
| if (PySequence_Fast_GET_SIZE(s_fast) == 0) { |
| Py_DECREF(s_fast); |
| return _PyAccu_Accumulate(acc, empty_array); |
| } |
| |
| if (s->markers != Py_None) { |
| int has_key; |
| ident = PyLong_FromVoidPtr(seq); |
| if (ident == NULL) |
| goto bail; |
| has_key = PyDict_Contains(s->markers, ident); |
| if (has_key) { |
| if (has_key != -1) |
| PyErr_SetString(PyExc_ValueError, "Circular reference detected"); |
| goto bail; |
| } |
| if (PyDict_SetItem(s->markers, ident, seq)) { |
| goto bail; |
| } |
| } |
| |
| if (_PyAccu_Accumulate(acc, open_array)) |
| goto bail; |
| if (s->indent != Py_None) { |
| /* TODO: DOES NOT RUN */ |
| indent_level += 1; |
| /* |
| newline_indent = '\n' + (' ' * (_indent * _current_indent_level)) |
| separator = _item_separator + newline_indent |
| buf += newline_indent |
| */ |
| } |
| for (i = 0; i < PySequence_Fast_GET_SIZE(s_fast); i++) { |
| PyObject *obj = PySequence_Fast_GET_ITEM(s_fast, i); |
| if (i) { |
| if (_PyAccu_Accumulate(acc, s->item_separator)) |
| goto bail; |
| } |
| if (encoder_listencode_obj(s, acc, obj, indent_level)) |
| goto bail; |
| } |
| if (ident != NULL) { |
| if (PyDict_DelItem(s->markers, ident)) |
| goto bail; |
| Py_CLEAR(ident); |
| } |
| |
| /* TODO: DOES NOT RUN |
| if (s->indent != Py_None) { |
| indent_level -= 1; |
| |
| yield '\n' + (' ' * (_indent * _current_indent_level)) |
| }*/ |
| if (_PyAccu_Accumulate(acc, close_array)) |
| goto bail; |
| Py_DECREF(s_fast); |
| return 0; |
| |
| bail: |
| Py_XDECREF(ident); |
| Py_DECREF(s_fast); |
| return -1; |
| } |
| |
| static void |
| encoder_dealloc(PyObject *self) |
| { |
| /* 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}, |
| {"encode_basestring", |
| (PyCFunction)py_encode_basestring, |
| METH_O, |
| pydoc_encode_basestring}, |
| {"scanstring", |
| (PyCFunction)py_scanstring, |
| METH_VARARGS, |
| pydoc_scanstring}, |
| {NULL, NULL, 0, NULL} |
| }; |
| |
| PyDoc_STRVAR(module_doc, |
| "json speedups\n"); |
| |
| static struct PyModuleDef jsonmodule = { |
| PyModuleDef_HEAD_INIT, |
| "_json", |
| module_doc, |
| -1, |
| speedups_methods, |
| NULL, |
| NULL, |
| NULL, |
| NULL |
| }; |
| |
| PyMODINIT_FUNC |
| PyInit__json(void) |
| { |
| PyObject *m = PyModule_Create(&jsonmodule); |
| if (!m) |
| return NULL; |
| 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; |
| } |