mirror of
https://github.com/python/cpython.git
synced 2025-08-04 08:59:19 +00:00
Issue #12791: Break reference cycles early when a generator exits with an exception.
This commit is contained in:
parent
0c17d0d96e
commit
a370fcf3b2
3 changed files with 76 additions and 0 deletions
|
@ -607,6 +607,68 @@ class ExceptionTests(unittest.TestCase):
|
|||
gc_collect()
|
||||
self.assertEqual(sys.exc_info(), (None, None, None))
|
||||
|
||||
def _check_generator_cleanup_exc_state(self, testfunc):
|
||||
# Issue #12791: exception state is cleaned up as soon as a generator
|
||||
# is closed (reference cycles are broken).
|
||||
class MyException(Exception):
|
||||
def __init__(self, obj):
|
||||
self.obj = obj
|
||||
class MyObj:
|
||||
pass
|
||||
|
||||
def raising_gen():
|
||||
try:
|
||||
raise MyException(obj)
|
||||
except MyException:
|
||||
yield
|
||||
|
||||
obj = MyObj()
|
||||
wr = weakref.ref(obj)
|
||||
g = raising_gen()
|
||||
next(g)
|
||||
testfunc(g)
|
||||
g = obj = None
|
||||
obj = wr()
|
||||
self.assertIs(obj, None)
|
||||
|
||||
def test_generator_throw_cleanup_exc_state(self):
|
||||
def do_throw(g):
|
||||
try:
|
||||
g.throw(RuntimeError())
|
||||
except RuntimeError:
|
||||
pass
|
||||
self._check_generator_cleanup_exc_state(do_throw)
|
||||
|
||||
def test_generator_close_cleanup_exc_state(self):
|
||||
def do_close(g):
|
||||
g.close()
|
||||
self._check_generator_cleanup_exc_state(do_close)
|
||||
|
||||
def test_generator_del_cleanup_exc_state(self):
|
||||
def do_del(g):
|
||||
g = None
|
||||
self._check_generator_cleanup_exc_state(do_del)
|
||||
|
||||
def test_generator_next_cleanup_exc_state(self):
|
||||
def do_next(g):
|
||||
try:
|
||||
next(g)
|
||||
except StopIteration:
|
||||
pass
|
||||
else:
|
||||
self.fail("should have raised StopIteration")
|
||||
self._check_generator_cleanup_exc_state(do_next)
|
||||
|
||||
def test_generator_send_cleanup_exc_state(self):
|
||||
def do_send(g):
|
||||
try:
|
||||
g.send(None)
|
||||
except StopIteration:
|
||||
pass
|
||||
else:
|
||||
self.fail("should have raised StopIteration")
|
||||
self._check_generator_cleanup_exc_state(do_send)
|
||||
|
||||
def test_3114(self):
|
||||
# Bug #3114: in its destructor, MyObject retrieves a pointer to
|
||||
# obsolete and/or deallocated objects.
|
||||
|
|
Loading…
Add table
Add a link
Reference in a new issue