Commit bc2aa816 authored by Victor Stinner's avatar Victor Stinner Committed by GitHub

bpo-18748: _pyio.IOBase emits unraisable exception (GH-13512)

In development (-X dev) mode and in a debug build, IOBase finalizer
of the _pyio module now logs the exception if the close() method
fails. The exception is ignored silently by default in release build.

test_io: test_error_through_destructor() now uses
support.catch_unraisable_exception() rather than capturing stderr.
parent 0a8e5724
...@@ -318,6 +318,15 @@ for :func:`property`, :func:`classmethod`, and :func:`staticmethod`:: ...@@ -318,6 +318,15 @@ for :func:`property`, :func:`classmethod`, and :func:`staticmethod`::
self.bit_rate = round(bit_rate / 1000.0, 1) self.bit_rate = round(bit_rate / 1000.0, 1)
self.duration = ceil(duration) self.duration = ceil(duration)
io
--
In development mode (:option:`-X` ``env``) and in debug build, the
:class:`io.IOBase` finalizer now logs the exception if the ``close()`` method
fails. The exception is ignored silently by default in release build.
(Contributed by Victor Stinner in :issue:`18748`.)
gc gc
-- --
......
...@@ -33,6 +33,10 @@ DEFAULT_BUFFER_SIZE = 8 * 1024 # bytes ...@@ -33,6 +33,10 @@ DEFAULT_BUFFER_SIZE = 8 * 1024 # bytes
# Rebind for compatibility # Rebind for compatibility
BlockingIOError = BlockingIOError BlockingIOError = BlockingIOError
# Does io.IOBase finalizer log the exception if the close() method fails?
# The exception is ignored silently by default in release build.
_IOBASE_EMITS_UNRAISABLE = (hasattr(sys, "gettotalrefcount") or sys.flags.dev_mode)
def open(file, mode="r", buffering=-1, encoding=None, errors=None, def open(file, mode="r", buffering=-1, encoding=None, errors=None,
newline=None, closefd=True, opener=None): newline=None, closefd=True, opener=None):
...@@ -378,15 +382,18 @@ class IOBase(metaclass=abc.ABCMeta): ...@@ -378,15 +382,18 @@ class IOBase(metaclass=abc.ABCMeta):
def __del__(self): def __del__(self):
"""Destructor. Calls close().""" """Destructor. Calls close()."""
# The try/except block is in case this is called at program if _IOBASE_EMITS_UNRAISABLE:
# exit time, when it's possible that globals have already been
# deleted, and then the close() call might fail. Since
# there's nothing we can do about such failures and they annoy
# the end users, we suppress the traceback.
try:
self.close() self.close()
except: else:
pass # The try/except block is in case this is called at program
# exit time, when it's possible that globals have already been
# deleted, and then the close() call might fail. Since
# there's nothing we can do about such failures and they annoy
# the end users, we suppress the traceback.
try:
self.close()
except:
pass
### Inquiries ### ### Inquiries ###
......
...@@ -67,9 +67,9 @@ MEMORY_SANITIZER = ( ...@@ -67,9 +67,9 @@ MEMORY_SANITIZER = (
'--with-memory-sanitizer' in _config_args '--with-memory-sanitizer' in _config_args
) )
# Does io.IOBase logs unhandled exceptions on calling close()? # Does io.IOBase finalizer log the exception if the close() method fails?
# They are silenced by default in release build. # The exception is ignored silently by default in release build.
DESTRUCTOR_LOG_ERRORS = (hasattr(sys, "gettotalrefcount") or sys.flags.dev_mode) IOBASE_EMITS_UNRAISABLE = (hasattr(sys, "gettotalrefcount") or sys.flags.dev_mode)
def _default_chunk_size(): def _default_chunk_size():
...@@ -1098,23 +1098,18 @@ class CommonBufferedTests: ...@@ -1098,23 +1098,18 @@ class CommonBufferedTests:
# Test that the exception state is not modified by a destructor, # Test that the exception state is not modified by a destructor,
# even if close() fails. # even if close() fails.
rawio = self.CloseFailureIO() rawio = self.CloseFailureIO()
def f(): try:
self.tp(rawio).xyzzy with support.catch_unraisable_exception() as cm:
with support.captured_output("stderr") as s: with self.assertRaises(AttributeError):
self.assertRaises(AttributeError, f) self.tp(rawio).xyzzy
s = s.getvalue().strip()
if s: if not IOBASE_EMITS_UNRAISABLE:
# The destructor *may* have printed an unraisable error, check it self.assertIsNone(cm.unraisable)
lines = s.splitlines() elif cm.unraisable is not None:
if DESTRUCTOR_LOG_ERRORS: self.assertEqual(cm.unraisable.exc_type, OSError)
self.assertEqual(len(lines), 5) finally:
self.assertTrue(lines[0].startswith("Exception ignored in: "), lines) # Explicitly break reference cycle
self.assertEqual(lines[1], "Traceback (most recent call last):", lines) cm = None
self.assertEqual(lines[4], 'OSError:', lines)
else:
self.assertEqual(len(lines), 1)
self.assertTrue(lines[-1].startswith("Exception OSError: "), lines)
self.assertTrue(lines[-1].endswith(" ignored"), lines)
def test_repr(self): def test_repr(self):
raw = self.MockRawIO() raw = self.MockRawIO()
...@@ -2859,23 +2854,18 @@ class TextIOWrapperTest(unittest.TestCase): ...@@ -2859,23 +2854,18 @@ class TextIOWrapperTest(unittest.TestCase):
# Test that the exception state is not modified by a destructor, # Test that the exception state is not modified by a destructor,
# even if close() fails. # even if close() fails.
rawio = self.CloseFailureIO() rawio = self.CloseFailureIO()
def f(): try:
self.TextIOWrapper(rawio).xyzzy with support.catch_unraisable_exception() as cm:
with support.captured_output("stderr") as s: with self.assertRaises(AttributeError):
self.assertRaises(AttributeError, f) self.TextIOWrapper(rawio).xyzzy
s = s.getvalue().strip()
if s: if not IOBASE_EMITS_UNRAISABLE:
# The destructor *may* have printed an unraisable error, check it self.assertIsNone(cm.unraisable)
lines = s.splitlines() elif cm.unraisable is not None:
if DESTRUCTOR_LOG_ERRORS: self.assertEqual(cm.unraisable.exc_type, OSError)
self.assertEqual(len(lines), 5) finally:
self.assertTrue(lines[0].startswith("Exception ignored in: "), lines) # Explicitly break reference cycle
self.assertEqual(lines[1], "Traceback (most recent call last):", lines) cm = None
self.assertEqual(lines[4], 'OSError:', lines)
else:
self.assertEqual(len(lines), 1)
self.assertTrue(lines[-1].startswith("Exception OSError: "), lines)
self.assertTrue(lines[-1].endswith(" ignored"), lines)
# Systematic tests of the text I/O API # Systematic tests of the text I/O API
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment