Skip to content

Commit 253279c

Browse files
authored
[2.7] closes bpo-31608: Fix a crash in methods of a subclass of _collections.deque with a bad __new__(). (GH-9179)
1 parent 7a501de commit 253279c

3 files changed

Lines changed: 28 additions & 2 deletions

File tree

Lib/test/test_deque.py

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

662+
@test_support.cpython_only
663+
def test_bug_31608(self):
664+
# The interpreter used to crash in specific cases where a deque
665+
# subclass returned a non-deque.
666+
class X(deque):
667+
pass
668+
d = X()
669+
def bad___new__(cls, *args, **kwargs):
670+
return [42]
671+
X.__new__ = bad___new__
672+
with self.assertRaises(TypeError):
673+
d * 42 # shouldn't crash
674+
with self.assertRaises(TypeError):
675+
d + deque([1, 2, 3]) # shouldn't crash
676+
662677

663678
class SubclassWithKwargs(deque):
664679
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: 11 additions & 2 deletions
Original file line numberDiff line numberDiff line change
@@ -859,11 +859,20 @@ deque_traverse(dequeobject *deque, visitproc visit, void *arg)
859859
static PyObject *
860860
deque_copy(PyObject *deque)
861861
{
862+
PyObject *result;
862863
if (((dequeobject *)deque)->maxlen == -1)
863-
return PyObject_CallFunction((PyObject *)(Py_TYPE(deque)), "O", deque, NULL);
864+
result = PyObject_CallFunction((PyObject *)(Py_TYPE(deque)), "O", deque, NULL);
864865
else
865-
return PyObject_CallFunction((PyObject *)(Py_TYPE(deque)), "Oi",
866+
result = PyObject_CallFunction((PyObject *)(Py_TYPE(deque)), "Oi",
866867
deque, ((dequeobject *)deque)->maxlen, NULL);
868+
if (result != NULL && !PyObject_TypeCheck(result, &deque_type)) {
869+
PyErr_Format(PyExc_TypeError,
870+
"%.200s() must return a deque, not %.200s",
871+
Py_TYPE(deque)->tp_name, Py_TYPE(result)->tp_name);
872+
Py_DECREF(result);
873+
return NULL;
874+
}
875+
return result;
867876
}
868877

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

0 commit comments

Comments
 (0)