Commit 02ab7a84 authored by Benjamin Peterson's avatar Benjamin Peterson

make sure fdopen always closes the fd in error cases (closes #21191)

parent d5460096
...@@ -463,8 +463,9 @@ These functions create new file objects. (See also :func:`open`.) ...@@ -463,8 +463,9 @@ These functions create new file objects. (See also :func:`open`.)
.. index:: single: I/O control; buffering .. index:: single: I/O control; buffering
Return an open file object connected to the file descriptor *fd*. The *mode* Return an open file object connected to the file descriptor *fd*. The *mode*
and *bufsize* arguments have the same meaning as the corresponding arguments to and *bufsize* arguments have the same meaning as the corresponding arguments
the built-in :func:`open` function. to the built-in :func:`open` function. If :func:`fdopen` raises an
exception, it closes *fd*.
Availability: Unix, Windows. Availability: Unix, Windows.
......
...@@ -194,6 +194,10 @@ class PosixTester(unittest.TestCase): ...@@ -194,6 +194,10 @@ class PosixTester(unittest.TestCase):
self.fdopen_helper('r') self.fdopen_helper('r')
self.fdopen_helper('r', 100) self.fdopen_helper('r', 100)
fd = os.open(test_support.TESTFN, os.O_RDONLY)
self.assertRaises(OSError, posix.fdopen, fd, 'w')
self.assertRaises(OSError, os.close, fd) # fd should be closed.
@unittest.skipUnless(hasattr(posix, 'O_EXLOCK'), @unittest.skipUnless(hasattr(posix, 'O_EXLOCK'),
'test needs posix.O_EXLOCK') 'test needs posix.O_EXLOCK')
def test_osexlock(self): def test_osexlock(self):
......
...@@ -43,6 +43,9 @@ Core and Builtins ...@@ -43,6 +43,9 @@ Core and Builtins
Library Library
------- -------
- Issue #21191: In os.fdopen, alwyas close the file descriptor when an exception
happens.
- Issue #21149: Improved thread-safety in logging cleanup during interpreter - Issue #21149: Improved thread-safety in logging cleanup during interpreter
shutdown. Thanks to Devin Jeanpierre for the patch. shutdown. Thanks to Devin Jeanpierre for the patch.
......
...@@ -6841,16 +6841,21 @@ posix_fdopen(PyObject *self, PyObject *args) ...@@ -6841,16 +6841,21 @@ posix_fdopen(PyObject *self, PyObject *args)
/* Sanitize mode. See fileobject.c */ /* Sanitize mode. See fileobject.c */
mode = PyMem_MALLOC(strlen(orgmode)+3); mode = PyMem_MALLOC(strlen(orgmode)+3);
if (!mode) { if (!mode) {
close(fd);
PyErr_NoMemory(); PyErr_NoMemory();
return NULL; return NULL;
} }
strcpy(mode, orgmode); strcpy(mode, orgmode);
if (_PyFile_SanitizeMode(mode)) { if (_PyFile_SanitizeMode(mode)) {
close(fd);
PyMem_FREE(mode); PyMem_FREE(mode);
return NULL; return NULL;
} }
if (!_PyVerify_fd(fd)) if (!_PyVerify_fd(fd)) {
return posix_error(); posix_error();
close(fd);
return NULL;
}
Py_BEGIN_ALLOW_THREADS Py_BEGIN_ALLOW_THREADS
#if !defined(MS_WINDOWS) && defined(HAVE_FCNTL_H) #if !defined(MS_WINDOWS) && defined(HAVE_FCNTL_H)
if (mode[0] == 'a') { if (mode[0] == 'a') {
...@@ -6871,8 +6876,11 @@ posix_fdopen(PyObject *self, PyObject *args) ...@@ -6871,8 +6876,11 @@ posix_fdopen(PyObject *self, PyObject *args)
#endif #endif
Py_END_ALLOW_THREADS Py_END_ALLOW_THREADS
PyMem_FREE(mode); PyMem_FREE(mode);
if (fp == NULL) if (fp == NULL) {
return posix_error(); posix_error();
close(fd);
return NULL;
}
/* The dummy filename used here must be kept in sync with the value /* The dummy filename used here must be kept in sync with the value
tested against in gzip.GzipFile.__init__() - see issue #13781. */ tested against in gzip.GzipFile.__init__() - see issue #13781. */
f = PyFile_FromFile(fp, "<fdopen>", orgmode, fclose); f = PyFile_FromFile(fp, "<fdopen>", orgmode, fclose);
......
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