diff options
author | Tim Golden <mail@timgolden.me.uk> | 2010-08-06 13:20:12 (GMT) |
---|---|---|
committer | Tim Golden <mail@timgolden.me.uk> | 2010-08-06 13:20:12 (GMT) |
commit | 40b3744efa2572ce429771d6971f7972a12c2405 (patch) | |
tree | 2df2b7c9efee984171de20b9d2d6997324fe0df9 | |
parent | 51ced7afe72d861b46f069d757f2787f13253d32 (diff) | |
download | cpython-40b3744efa2572ce429771d6971f7972a12c2405.zip cpython-40b3744efa2572ce429771d6971f7972a12c2405.tar.gz cpython-40b3744efa2572ce429771d6971f7972a12c2405.tar.bz2 |
Issue #3210: Ensure stdio handles are closed if CreateProcess fails
-rw-r--r-- | Lib/test/test_subprocess.py | 20 | ||||
-rw-r--r-- | PC/_subprocess.c | 19 |
2 files changed, 37 insertions, 2 deletions
diff --git a/Lib/test/test_subprocess.py b/Lib/test/test_subprocess.py index 5cdbe2d..ff9945b 100644 --- a/Lib/test/test_subprocess.py +++ b/Lib/test/test_subprocess.py @@ -548,6 +548,26 @@ class ProcessTestCase(unittest.TestCase): output = subprocess.check_output([sys.executable, '-c', code]) self.assert_(output.startswith(b'Hello World!'), ascii(output)) + def test_handles_closed_on_exception(self): + # If CreateProcess exits with an error, ensure the + # duplicate output handles are released + ifhandle, ifname = self.mkstemp() + ofhandle, ofname = self.mkstemp() + efhandle, efname = self.mkstemp() + try: + subprocess.Popen (["*"], stdin=ifhandle, stdout=ofhandle, + stderr=efhandle) + except OSError: + os.close(ifhandle) + os.remove(ifname) + os.close(ofhandle) + os.remove(ofname) + os.close(efhandle) + os.remove(efname) + self.assertFalse(os.path.exists(ifname)) + self.assertFalse(os.path.exists(ofname)) + self.assertFalse(os.path.exists(efname)) + # # POSIX tests # diff --git a/PC/_subprocess.c b/PC/_subprocess.c index 27d3dc6..635abc8 100644 --- a/PC/_subprocess.c +++ b/PC/_subprocess.c @@ -429,6 +429,7 @@ sp_CreateProcess(PyObject* self, PyObject* args) PyObject* env_mapping; Py_UNICODE* current_directory; PyObject* startup_info; + DWORD error; if (! PyArg_ParseTuple(args, "ZZOOiiOZO:CreateProcess", &application_name, @@ -478,8 +479,22 @@ sp_CreateProcess(PyObject* self, PyObject* args) Py_XDECREF(environment); - if (! result) - return PyErr_SetFromWindowsErr(GetLastError()); + if (! result) { + error = GetLastError(); + if(si.hStdInput != INVALID_HANDLE_VALUE) { + CloseHandle(si.hStdInput); + si.hStdInput = INVALID_HANDLE_VALUE; + } + if(si.hStdOutput != INVALID_HANDLE_VALUE) { + CloseHandle(si.hStdOutput); + si.hStdOutput = INVALID_HANDLE_VALUE; + } + if(si.hStdError != INVALID_HANDLE_VALUE) { + CloseHandle(si.hStdError); + si.hStdError = INVALID_HANDLE_VALUE; + } + return PyErr_SetFromWindowsErr(error); + } return Py_BuildValue("NNii", sp_handle_new(pi.hProcess), |