Skip to content

Commit

Permalink
bpo-44720: Don't crash when calling weakref.proxy(not_an_iterator).__…
Browse files Browse the repository at this point in the history
…next__ (GH-27316) (#27325)

(cherry picked from commit 5370f0a)

Co-authored-by: Dennis Sweeney <36520290+sweeneyde@users.noreply.github.com>
  • Loading branch information
miss-islington and sweeneyde committed Jul 24, 2021
1 parent 0978018 commit 0a08f22
Show file tree
Hide file tree
Showing 3 changed files with 37 additions and 0 deletions.
30 changes: 30 additions & 0 deletions Lib/test/test_weakref.py
Expand Up @@ -411,6 +411,36 @@ def __iter__(self):
# can be killed in the middle of the call
"blech" in p

def test_proxy_next(self):
arr = [4, 5, 6]
def iterator_func():
yield from arr
it = iterator_func()

class IteratesWeakly:
def __iter__(self):
return weakref.proxy(it)

weak_it = IteratesWeakly()

# Calls proxy.__next__
self.assertEqual(list(weak_it), [4, 5, 6])

def test_proxy_bad_next(self):
# bpo-44720: PyIter_Next() shouldn't be called if the reference
# isn't an iterator.

not_an_iterator = lambda: 0

class A:
def __iter__(self):
return weakref.proxy(not_an_iterator)
a = A()

msg = "Weakref proxy referenced a non-iterator"
with self.assertRaisesRegex(TypeError, msg):
list(a)

def test_proxy_reversed(self):
class MyObj:
def __len__(self):
Expand Down
@@ -0,0 +1 @@
``weakref.proxy`` objects referencing non-iterators now raise ``TypeError`` rather than dereferencing the null ``tp_iternext`` slot and crashing.
6 changes: 6 additions & 0 deletions Objects/weakrefobject.c
Expand Up @@ -657,6 +657,12 @@ proxy_iternext(PyWeakReference *proxy)
return NULL;

PyObject *obj = PyWeakref_GET_OBJECT(proxy);
if (!PyIter_Check(obj)) {
PyErr_Format(PyExc_TypeError,
"Weakref proxy referenced a non-iterator '%.200s' object",
Py_TYPE(obj)->tp_name);
return NULL;
}
Py_INCREF(obj);
PyObject* res = PyIter_Next(obj);
Py_DECREF(obj);
Expand Down

0 comments on commit 0a08f22

Please sign in to comment.