| #include "parts.h" |
| #include "util.h" |
| |
| |
| static PyObject * |
| dict_check(PyObject *self, PyObject *obj) |
| { |
| NULLABLE(obj); |
| return PyLong_FromLong(PyDict_Check(obj)); |
| } |
| |
| static PyObject * |
| dict_checkexact(PyObject *self, PyObject *obj) |
| { |
| NULLABLE(obj); |
| return PyLong_FromLong(PyDict_CheckExact(obj)); |
| } |
| |
| static PyObject * |
| dict_new(PyObject *self, PyObject *Py_UNUSED(ignored)) |
| { |
| return PyDict_New(); |
| } |
| |
| static PyObject * |
| dictproxy_new(PyObject *self, PyObject *obj) |
| { |
| NULLABLE(obj); |
| return PyDictProxy_New(obj); |
| } |
| |
| static PyObject * |
| dict_clear(PyObject *self, PyObject *obj) |
| { |
| PyDict_Clear(obj); |
| Py_RETURN_NONE; |
| } |
| |
| static PyObject * |
| dict_copy(PyObject *self, PyObject *obj) |
| { |
| NULLABLE(obj); |
| return PyDict_Copy(obj); |
| } |
| |
| static PyObject * |
| dict_contains(PyObject *self, PyObject *args) |
| { |
| PyObject *obj, *key; |
| if (!PyArg_ParseTuple(args, "OO", &obj, &key)) { |
| return NULL; |
| } |
| NULLABLE(obj); |
| NULLABLE(key); |
| RETURN_INT(PyDict_Contains(obj, key)); |
| } |
| |
| static PyObject * |
| dict_containsstring(PyObject *self, PyObject *args) |
| { |
| PyObject *obj; |
| const char *key; |
| Py_ssize_t size; |
| if (!PyArg_ParseTuple(args, "Oz#", &obj, &key, &size)) { |
| return NULL; |
| } |
| NULLABLE(obj); |
| RETURN_INT(PyDict_ContainsString(obj, key)); |
| } |
| |
| static PyObject * |
| dict_size(PyObject *self, PyObject *obj) |
| { |
| NULLABLE(obj); |
| RETURN_SIZE(PyDict_Size(obj)); |
| } |
| |
| static PyObject * |
| dict_getitem(PyObject *self, PyObject *args) |
| { |
| PyObject *mapping, *key; |
| if (!PyArg_ParseTuple(args, "OO", &mapping, &key)) { |
| return NULL; |
| } |
| NULLABLE(mapping); |
| NULLABLE(key); |
| PyObject *value = PyDict_GetItem(mapping, key); |
| if (value == NULL) { |
| if (PyErr_Occurred()) { |
| return NULL; |
| } |
| return Py_NewRef(PyExc_KeyError); |
| } |
| return Py_NewRef(value); |
| } |
| |
| static PyObject * |
| dict_getitemstring(PyObject *self, PyObject *args) |
| { |
| PyObject *mapping; |
| const char *key; |
| Py_ssize_t size; |
| if (!PyArg_ParseTuple(args, "Oz#", &mapping, &key, &size)) { |
| return NULL; |
| } |
| NULLABLE(mapping); |
| PyObject *value = PyDict_GetItemString(mapping, key); |
| if (value == NULL) { |
| if (PyErr_Occurred()) { |
| return NULL; |
| } |
| return Py_NewRef(PyExc_KeyError); |
| } |
| return Py_NewRef(value); |
| } |
| |
| static PyObject * |
| dict_getitemwitherror(PyObject *self, PyObject *args) |
| { |
| PyObject *mapping, *key; |
| if (!PyArg_ParseTuple(args, "OO", &mapping, &key)) { |
| return NULL; |
| } |
| NULLABLE(mapping); |
| NULLABLE(key); |
| PyObject *value = PyDict_GetItemWithError(mapping, key); |
| if (value == NULL) { |
| if (PyErr_Occurred()) { |
| return NULL; |
| } |
| return Py_NewRef(PyExc_KeyError); |
| } |
| return Py_NewRef(value); |
| } |
| |
| |
| static PyObject * |
| dict_getitemref(PyObject *self, PyObject *args) |
| { |
| PyObject *obj, *attr_name, *value = UNINITIALIZED_PTR; |
| if (!PyArg_ParseTuple(args, "OO", &obj, &attr_name)) { |
| return NULL; |
| } |
| NULLABLE(obj); |
| NULLABLE(attr_name); |
| |
| switch (PyDict_GetItemRef(obj, attr_name, &value)) { |
| case -1: |
| assert(value == NULL); |
| return NULL; |
| case 0: |
| assert(value == NULL); |
| return Py_NewRef(PyExc_KeyError); |
| case 1: |
| return value; |
| default: |
| Py_FatalError("PyMapping_GetItemRef() returned invalid code"); |
| Py_UNREACHABLE(); |
| } |
| } |
| |
| static PyObject * |
| dict_getitemstringref(PyObject *self, PyObject *args) |
| { |
| PyObject *obj, *value = UNINITIALIZED_PTR; |
| const char *attr_name; |
| Py_ssize_t size; |
| if (!PyArg_ParseTuple(args, "Oz#", &obj, &attr_name, &size)) { |
| return NULL; |
| } |
| NULLABLE(obj); |
| |
| switch (PyDict_GetItemStringRef(obj, attr_name, &value)) { |
| case -1: |
| assert(value == NULL); |
| return NULL; |
| case 0: |
| assert(value == NULL); |
| return Py_NewRef(PyExc_KeyError); |
| case 1: |
| return value; |
| default: |
| Py_FatalError("PyDict_GetItemStringRef() returned invalid code"); |
| Py_UNREACHABLE(); |
| } |
| } |
| |
| static PyObject * |
| dict_setitem(PyObject *self, PyObject *args) |
| { |
| PyObject *mapping, *key, *value; |
| if (!PyArg_ParseTuple(args, "OOO", &mapping, &key, &value)) { |
| return NULL; |
| } |
| NULLABLE(mapping); |
| NULLABLE(key); |
| NULLABLE(value); |
| RETURN_INT(PyDict_SetItem(mapping, key, value)); |
| } |
| |
| static PyObject * |
| dict_setitemstring(PyObject *self, PyObject *args) |
| { |
| PyObject *mapping, *value; |
| const char *key; |
| Py_ssize_t size; |
| if (!PyArg_ParseTuple(args, "Oz#O", &mapping, &key, &size, &value)) { |
| return NULL; |
| } |
| NULLABLE(mapping); |
| NULLABLE(value); |
| RETURN_INT(PyDict_SetItemString(mapping, key, value)); |
| } |
| |
| static PyObject * |
| dict_setdefault(PyObject *self, PyObject *args) |
| { |
| PyObject *mapping, *key, *defaultobj; |
| if (!PyArg_ParseTuple(args, "OOO", &mapping, &key, &defaultobj)) { |
| return NULL; |
| } |
| NULLABLE(mapping); |
| NULLABLE(key); |
| NULLABLE(defaultobj); |
| return PyDict_SetDefault(mapping, key, defaultobj); |
| } |
| |
| static PyObject * |
| dict_setdefaultref(PyObject *self, PyObject *args) |
| { |
| PyObject *obj, *key, *default_value, *result = UNINITIALIZED_PTR; |
| if (!PyArg_ParseTuple(args, "OOO", &obj, &key, &default_value)) { |
| return NULL; |
| } |
| NULLABLE(obj); |
| NULLABLE(key); |
| NULLABLE(default_value); |
| switch (PyDict_SetDefaultRef(obj, key, default_value, &result)) { |
| case -1: |
| assert(result == NULL); |
| return NULL; |
| case 0: |
| assert(result == default_value); |
| return result; |
| case 1: |
| return result; |
| default: |
| Py_FatalError("PyDict_SetDefaultRef() returned invalid code"); |
| Py_UNREACHABLE(); |
| } |
| } |
| |
| static PyObject * |
| dict_delitem(PyObject *self, PyObject *args) |
| { |
| PyObject *mapping, *key; |
| if (!PyArg_ParseTuple(args, "OO", &mapping, &key)) { |
| return NULL; |
| } |
| NULLABLE(mapping); |
| NULLABLE(key); |
| RETURN_INT(PyDict_DelItem(mapping, key)); |
| } |
| |
| static PyObject * |
| dict_delitemstring(PyObject *self, PyObject *args) |
| { |
| PyObject *mapping; |
| const char *key; |
| Py_ssize_t size; |
| if (!PyArg_ParseTuple(args, "Oz#", &mapping, &key, &size)) { |
| return NULL; |
| } |
| NULLABLE(mapping); |
| RETURN_INT(PyDict_DelItemString(mapping, key)); |
| } |
| |
| static PyObject * |
| dict_keys(PyObject *self, PyObject *obj) |
| { |
| NULLABLE(obj); |
| return PyDict_Keys(obj); |
| } |
| |
| static PyObject * |
| dict_values(PyObject *self, PyObject *obj) |
| { |
| NULLABLE(obj); |
| return PyDict_Values(obj); |
| } |
| |
| static PyObject * |
| dict_items(PyObject *self, PyObject *obj) |
| { |
| NULLABLE(obj); |
| return PyDict_Items(obj); |
| } |
| |
| static PyObject * |
| dict_next(PyObject *self, PyObject *args) |
| { |
| PyObject *mapping, *key = UNINITIALIZED_PTR, *value = UNINITIALIZED_PTR; |
| Py_ssize_t pos; |
| if (!PyArg_ParseTuple(args, "On", &mapping, &pos)) { |
| return NULL; |
| } |
| NULLABLE(mapping); |
| int rc = PyDict_Next(mapping, &pos, &key, &value); |
| if (rc != 0) { |
| return Py_BuildValue("inOO", rc, pos, key, value); |
| } |
| assert(key == UNINITIALIZED_PTR); |
| assert(value == UNINITIALIZED_PTR); |
| if (PyErr_Occurred()) { |
| return NULL; |
| } |
| Py_RETURN_NONE; |
| } |
| |
| static PyObject * |
| dict_merge(PyObject *self, PyObject *args) |
| { |
| PyObject *mapping, *mapping2; |
| int override; |
| if (!PyArg_ParseTuple(args, "OOi", &mapping, &mapping2, &override)) { |
| return NULL; |
| } |
| NULLABLE(mapping); |
| NULLABLE(mapping2); |
| RETURN_INT(PyDict_Merge(mapping, mapping2, override)); |
| } |
| |
| static PyObject * |
| dict_update(PyObject *self, PyObject *args) |
| { |
| PyObject *mapping, *mapping2; |
| if (!PyArg_ParseTuple(args, "OO", &mapping, &mapping2)) { |
| return NULL; |
| } |
| NULLABLE(mapping); |
| NULLABLE(mapping2); |
| RETURN_INT(PyDict_Update(mapping, mapping2)); |
| } |
| |
| static PyObject * |
| dict_mergefromseq2(PyObject *self, PyObject *args) |
| { |
| PyObject *mapping, *seq; |
| int override; |
| if (!PyArg_ParseTuple(args, "OOi", &mapping, &seq, &override)) { |
| return NULL; |
| } |
| NULLABLE(mapping); |
| NULLABLE(seq); |
| RETURN_INT(PyDict_MergeFromSeq2(mapping, seq, override)); |
| } |
| |
| |
| static PyObject * |
| dict_pop(PyObject *self, PyObject *args) |
| { |
| // Test PyDict_Pop(dict, key, &value) |
| PyObject *dict, *key; |
| if (!PyArg_ParseTuple(args, "OO", &dict, &key)) { |
| return NULL; |
| } |
| NULLABLE(dict); |
| NULLABLE(key); |
| PyObject *result = UNINITIALIZED_PTR; |
| int res = PyDict_Pop(dict, key, &result); |
| if (res < 0) { |
| assert(result == NULL); |
| return NULL; |
| } |
| if (res == 0) { |
| assert(result == NULL); |
| result = Py_NewRef(Py_None); |
| } |
| else { |
| assert(result != NULL); |
| } |
| return Py_BuildValue("iN", res, result); |
| } |
| |
| |
| static PyObject * |
| dict_pop_null(PyObject *self, PyObject *args) |
| { |
| // Test PyDict_Pop(dict, key, NULL) |
| PyObject *dict, *key; |
| if (!PyArg_ParseTuple(args, "OO", &dict, &key)) { |
| return NULL; |
| } |
| NULLABLE(dict); |
| NULLABLE(key); |
| RETURN_INT(PyDict_Pop(dict, key, NULL)); |
| } |
| |
| |
| static PyObject * |
| dict_popstring(PyObject *self, PyObject *args) |
| { |
| PyObject *dict; |
| const char *key; |
| Py_ssize_t key_size; |
| if (!PyArg_ParseTuple(args, "Oz#", &dict, &key, &key_size)) { |
| return NULL; |
| } |
| NULLABLE(dict); |
| PyObject *result = UNINITIALIZED_PTR; |
| int res = PyDict_PopString(dict, key, &result); |
| if (res < 0) { |
| assert(result == NULL); |
| return NULL; |
| } |
| if (res == 0) { |
| assert(result == NULL); |
| result = Py_NewRef(Py_None); |
| } |
| else { |
| assert(result != NULL); |
| } |
| return Py_BuildValue("iN", res, result); |
| } |
| |
| |
| static PyObject * |
| dict_popstring_null(PyObject *self, PyObject *args) |
| { |
| PyObject *dict; |
| const char *key; |
| Py_ssize_t key_size; |
| if (!PyArg_ParseTuple(args, "Oz#", &dict, &key, &key_size)) { |
| return NULL; |
| } |
| NULLABLE(dict); |
| RETURN_INT(PyDict_PopString(dict, key, NULL)); |
| } |
| |
| |
| static PyMethodDef test_methods[] = { |
| {"dict_check", dict_check, METH_O}, |
| {"dict_checkexact", dict_checkexact, METH_O}, |
| {"dict_new", dict_new, METH_NOARGS}, |
| {"dictproxy_new", dictproxy_new, METH_O}, |
| {"dict_clear", dict_clear, METH_O}, |
| {"dict_copy", dict_copy, METH_O}, |
| {"dict_size", dict_size, METH_O}, |
| {"dict_getitem", dict_getitem, METH_VARARGS}, |
| {"dict_getitemwitherror", dict_getitemwitherror, METH_VARARGS}, |
| {"dict_getitemstring", dict_getitemstring, METH_VARARGS}, |
| {"dict_getitemref", dict_getitemref, METH_VARARGS}, |
| {"dict_getitemstringref", dict_getitemstringref, METH_VARARGS}, |
| {"dict_contains", dict_contains, METH_VARARGS}, |
| {"dict_containsstring", dict_containsstring, METH_VARARGS}, |
| {"dict_setitem", dict_setitem, METH_VARARGS}, |
| {"dict_setitemstring", dict_setitemstring, METH_VARARGS}, |
| {"dict_delitem", dict_delitem, METH_VARARGS}, |
| {"dict_delitemstring", dict_delitemstring, METH_VARARGS}, |
| {"dict_setdefault", dict_setdefault, METH_VARARGS}, |
| {"dict_setdefaultref", dict_setdefaultref, METH_VARARGS}, |
| {"dict_keys", dict_keys, METH_O}, |
| {"dict_values", dict_values, METH_O}, |
| {"dict_items", dict_items, METH_O}, |
| {"dict_next", dict_next, METH_VARARGS}, |
| {"dict_merge", dict_merge, METH_VARARGS}, |
| {"dict_update", dict_update, METH_VARARGS}, |
| {"dict_mergefromseq2", dict_mergefromseq2, METH_VARARGS}, |
| {"dict_pop", dict_pop, METH_VARARGS}, |
| {"dict_pop_null", dict_pop_null, METH_VARARGS}, |
| {"dict_popstring", dict_popstring, METH_VARARGS}, |
| {"dict_popstring_null", dict_popstring_null, METH_VARARGS}, |
| {NULL}, |
| }; |
| |
| int |
| _PyTestCapi_Init_Dict(PyObject *m) |
| { |
| if (PyModule_AddFunctions(m, test_methods) < 0) { |
| return -1; |
| } |
| |
| return 0; |
| } |