Skip to content

Commit 24bd50b

Browse files
orenmnbenjaminp
authored andcommitted
closes bpo-31608: Fix a crash in methods of a subclass of _collections.deque with a bad __new__(). (GH-3788)
1 parent b4ec362 commit 24bd50b

File tree

3 files changed

+30
-4
lines changed

3 files changed

+30
-4
lines changed

Lib/test/test_deque.py

Lines changed: 15 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -892,6 +892,21 @@ def __iter__(self):
892892
d1 == d2 # not clear if this is supposed to be True or False,
893893
# but it used to give a SystemError
894894

895+
@support.cpython_only
896+
def test_bug_31608(self):
897+
# The interpreter used to crash in specific cases where a deque
898+
# subclass returned a non-deque.
899+
class X(deque):
900+
pass
901+
d = X()
902+
def bad___new__(cls, *args, **kwargs):
903+
return [42]
904+
X.__new__ = bad___new__
905+
with self.assertRaises(TypeError):
906+
d * 42 # shouldn't crash
907+
with self.assertRaises(TypeError):
908+
d + deque([1, 2, 3]) # shouldn't crash
909+
895910

896911
class SubclassWithKwargs(deque):
897912
def __init__(self, newarg=1):
Lines changed: 2 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,2 @@
1+
Raise a ``TypeError`` instead of crashing if a ``collections.deque`` subclass
2+
returns a non-deque from ``__new__``. Patch by Oren Milman.

Modules/_collectionsmodule.c

Lines changed: 13 additions & 4 deletions
Original file line numberDiff line numberDiff line change
@@ -478,6 +478,7 @@ deque_inplace_concat(dequeobject *deque, PyObject *other)
478478
static PyObject *
479479
deque_copy(PyObject *deque, PyObject *Py_UNUSED(ignored))
480480
{
481+
PyObject *result;
481482
dequeobject *old_deque = (dequeobject *)deque;
482483
if (Py_TYPE(deque) == &deque_type) {
483484
dequeobject *new_deque;
@@ -502,11 +503,19 @@ deque_copy(PyObject *deque, PyObject *Py_UNUSED(ignored))
502503
return NULL;
503504
}
504505
if (old_deque->maxlen < 0)
505-
return PyObject_CallFunctionObjArgs((PyObject *)(Py_TYPE(deque)),
506-
deque, NULL);
506+
result = PyObject_CallFunctionObjArgs((PyObject *)(Py_TYPE(deque)),
507+
deque, NULL);
507508
else
508-
return PyObject_CallFunction((PyObject *)(Py_TYPE(deque)), "Oi",
509-
deque, old_deque->maxlen, NULL);
509+
result = PyObject_CallFunction((PyObject *)(Py_TYPE(deque)), "Oi",
510+
deque, old_deque->maxlen, NULL);
511+
if (result != NULL && !PyObject_TypeCheck(result, &deque_type)) {
512+
PyErr_Format(PyExc_TypeError,
513+
"%.200s() must return a deque, not %.200s",
514+
Py_TYPE(deque)->tp_name, Py_TYPE(result)->tp_name);
515+
Py_DECREF(result);
516+
return NULL;
517+
}
518+
return result;
510519
}
511520

512521
PyDoc_STRVAR(copy_doc, "Return a shallow copy of a deque.");

0 commit comments

Comments
 (0)