#10053: Don't close FDs when FileIO.__init__ fails

Loosely based on the work by Hirokazu Yamamoto.
This commit is contained in:
Hynek Schlawack 2012-06-21 19:45:19 +02:00
parent 4215d2738a
commit 9bd4bf2a3d
3 changed files with 20 additions and 6 deletions

View File

@ -421,6 +421,17 @@ class OtherFileTests(unittest.TestCase):
'IOError: [Errno 2] No such file or directory' not in out):
self.fail('Bad output: %r' % out)
def testUnclosedFDOnException(self):
class MyException(Exception): pass
class MyFileIO(_FileIO):
def __setattr__(self, name, value):
if name == "name":
raise MyException("blocked setting name")
return super(MyFileIO, self).__setattr__(name, value)
fd = os.open(__file__, os.O_RDONLY)
self.assertRaises(MyException, MyFileIO, fd)
os.close(fd) # should not raise OSError(EBADF)
def test_main():
# Historically, these tests have been sloppy about removing TESTFN.
# So get rid of it no matter what.

View File

@ -9,6 +9,9 @@ What's New in Python 2.7.4
Core and Builtins
-----------------
- Issue #10053: Don't close FDs when FileIO.__init__ fails. Loosely based on
the work by Hirokazu Yamamoto.
- Issue #14775: Fix a potential quadratic dict build-up due to the garbage
collector repeatedly trying to untrack dicts.

View File

@ -195,6 +195,7 @@ fileio_init(PyObject *oself, PyObject *args, PyObject *kwds)
int flags = 0;
int fd = -1;
int closefd = 1;
int fd_is_own = 0;
assert(PyFileIO_Check(oself));
if (self->fd >= 0) {
@ -345,6 +346,7 @@ fileio_init(PyObject *oself, PyObject *args, PyObject *kwds)
#endif
self->fd = open(name, flags, 0666);
Py_END_ALLOW_THREADS
fd_is_own = 1;
if (self->fd < 0) {
#ifdef MS_WINDOWS
if (widename != NULL)
@ -366,19 +368,17 @@ fileio_init(PyObject *oself, PyObject *args, PyObject *kwds)
end of file (otherwise, it might be done only on the
first write()). */
PyObject *pos = portable_lseek(self->fd, NULL, 2);
if (pos == NULL) {
if (closefd) {
close(self->fd);
self->fd = -1;
}
if (pos == NULL)
goto error;
}
Py_DECREF(pos);
}
goto done;
error:
if (!fd_is_own)
self->fd = -1;
ret = -1;
done: