diff options
author | Benjamin Peterson <benjamin@python.org> | 2012-12-20 18:24:10 (GMT) |
---|---|---|
committer | Benjamin Peterson <benjamin@python.org> | 2012-12-20 18:24:10 (GMT) |
commit | a2d6d7121e1e6c7484003ddaf2f9fbe1a32a5a0e (patch) | |
tree | 208a37334cf0d2a4da15c94959ac98f19826a95d | |
parent | bacf1bf35502931ce5de8c2e026ab6be65f943ef (diff) | |
download | cpython-a2d6d7121e1e6c7484003ddaf2f9fbe1a32a5a0e.zip cpython-a2d6d7121e1e6c7484003ddaf2f9fbe1a32a5a0e.tar.gz cpython-a2d6d7121e1e6c7484003ddaf2f9fbe1a32a5a0e.tar.bz2 |
call close on the underlying stream even if flush raises (#16597)
-rw-r--r-- | Lib/_pyio.py | 12 | ||||
-rw-r--r-- | Lib/test/test_io.py | 27 | ||||
-rw-r--r-- | Modules/_io/bufferedio.c | 22 | ||||
-rw-r--r-- | Modules/_io/textio.c | 20 |
4 files changed, 68 insertions, 13 deletions
diff --git a/Lib/_pyio.py b/Lib/_pyio.py index 86f3c4c..ac7832e 100644 --- a/Lib/_pyio.py +++ b/Lib/_pyio.py @@ -340,8 +340,10 @@ class IOBase: This method has no effect if the file is already closed. """ if not self.__closed: - self.flush() - self.__closed = True + try: + self.flush() + finally: + self.__closed = True def __del__(self): """Destructor. Calls close().""" @@ -1568,8 +1570,10 @@ class TextIOWrapper(TextIOBase): def close(self): if self.buffer is not None and not self.closed: - self.flush() - self.buffer.close() + try: + self.flush() + finally: + self.buffer.close() @property def closed(self): diff --git a/Lib/test/test_io.py b/Lib/test/test_io.py index d10f52a..6132bae 100644 --- a/Lib/test/test_io.py +++ b/Lib/test/test_io.py @@ -573,6 +573,7 @@ class IOTest(unittest.TestCase): raise IOError() f.flush = bad_flush self.assertRaises(IOError, f.close) # exception not swallowed + self.assertTrue(f.closed) def test_multi_close(self): f = self.open(support.TESTFN, "wb", buffering=0) @@ -732,6 +733,21 @@ class CommonBufferedTests: raw.flush = bad_flush b = self.tp(raw) self.assertRaises(IOError, b.close) # exception not swallowed + self.assertTrue(b.closed) + + def test_close_error_on_close(self): + raw = self.MockRawIO() + def bad_flush(): + raise IOError('flush') + def bad_close(): + raise IOError('close') + raw.close = bad_close + b = self.tp(raw) + b.flush = bad_flush + with self.assertRaises(IOError) as err: # exception not swallowed + b.close() + self.assertEqual(err.exception.args, ('close',)) + self.assertFalse(b.closed) def test_multi_close(self): raw = self.MockRawIO() @@ -1230,6 +1246,16 @@ class BufferedWriterTest(unittest.TestCase, CommonBufferedTests): DeprecationWarning)): self.tp(self.MockRawIO(), 8, 12) + def test_write_error_on_close(self): + raw = self.MockRawIO() + def bad_write(b): + raise IOError() + raw.write = bad_write + b = self.tp(raw) + b.write(b'spam') + self.assertRaises(IOError, b.close) # exception not swallowed + self.assertTrue(b.closed) + class CBufferedWriterTest(BufferedWriterTest, SizeofTest): tp = io.BufferedWriter @@ -2378,6 +2404,7 @@ class TextIOWrapperTest(unittest.TestCase): raise IOError() txt.flush = bad_flush self.assertRaises(IOError, txt.close) # exception not swallowed + self.assertTrue(txt.closed) def test_multi_close(self): txt = self.TextIOWrapper(self.BytesIO(self.testdata), encoding="ascii") diff --git a/Modules/_io/bufferedio.c b/Modules/_io/bufferedio.c index 019ec94..e2d6025 100644 --- a/Modules/_io/bufferedio.c +++ b/Modules/_io/bufferedio.c @@ -455,7 +455,7 @@ buffered_closed_get(buffered *self, void *context) static PyObject * buffered_close(buffered *self, PyObject *args) { - PyObject *res = NULL; + PyObject *res = NULL, *exc = NULL, *val, *tb; int r; CHECK_INITIALIZED(self) @@ -475,13 +475,25 @@ buffered_close(buffered *self, PyObject *args) res = PyObject_CallMethodObjArgs((PyObject *)self, _PyIO_str_flush, NULL); if (!ENTER_BUFFERED(self)) return NULL; - if (res == NULL) { - goto end; - } - Py_XDECREF(res); + if (res == NULL) + PyErr_Fetch(&exc, &val, &tb); + else + Py_DECREF(res); res = PyObject_CallMethodObjArgs(self->raw, _PyIO_str_close, NULL); + if (exc != NULL) { + if (res != NULL) { + Py_CLEAR(res); + PyErr_Restore(exc, val, tb); + } + else { + Py_DECREF(exc); + Py_XDECREF(val); + Py_XDECREF(tb); + } + } + end: LEAVE_BUFFERED(self) return res; diff --git a/Modules/_io/textio.c b/Modules/_io/textio.c index e55c165..d2e92fa 100644 --- a/Modules/_io/textio.c +++ b/Modules/_io/textio.c @@ -2442,14 +2442,26 @@ textiowrapper_close(textio *self, PyObject *args) Py_RETURN_NONE; /* stream already closed */ } else { + PyObject *exc = NULL, *val, *tb; res = PyObject_CallMethod((PyObject *)self, "flush", NULL); - if (res == NULL) { - return NULL; - } + if (res == NULL) + PyErr_Fetch(&exc, &val, &tb); else Py_DECREF(res); - return PyObject_CallMethod(self->buffer, "close", NULL); + res = PyObject_CallMethod(self->buffer, "close", NULL); + if (exc != NULL) { + if (res != NULL) { + Py_CLEAR(res); + PyErr_Restore(exc, val, tb); + } + else { + Py_DECREF(exc); + Py_XDECREF(val); + Py_XDECREF(tb); + } + } + return res; } } |