]> granicus.if.org Git - python/commitdiff
#5179: don't leak PIPE fds when child execution fails.
authorGeorg Brandl <georg@python.org>
Sat, 14 Feb 2009 17:01:36 +0000 (17:01 +0000)
committerGeorg Brandl <georg@python.org>
Sat, 14 Feb 2009 17:01:36 +0000 (17:01 +0000)
Lib/subprocess.py
Lib/test/test_subprocess.py

index 309f5d4d83087c285648be748ae65f3aa53b667e..d91fbee34cef5aaa35692da951b070d898dbdf59 100644 (file)
@@ -1151,6 +1151,9 @@ class Popen(object):
             if data != "":
                 os.waitpid(self.pid, 0)
                 child_exception = pickle.loads(data)
+                for fd in (p2cwrite, c2pread, errread):
+                    if fd is not None:
+                        os.close(fd)
                 raise child_exception
 
 
index ae8e88877f0bf3c0c7d6490256770c3da0df6a3c..0ddac3dcb9d4fb28de138efe276c51e48b20f5cd 100644 (file)
@@ -520,6 +520,22 @@ class ProcessTestCase(unittest.TestCase):
         else:
             self.fail("Expected TypeError")
 
+    def test_leaking_fds_on_error(self):
+        # see bug #5179: Popen leaks file descriptors to PIPEs if
+        # the child fails to execute; this will eventually exhaust
+        # the maximum number of open fds. 1024 seems a very common
+        # value for that limit, but Windows has 2048, so we loop
+        # 1024 times (each call leaked two fds).
+        for i in range(1024):
+            try:
+                subprocess.Popen(['nonexisting_i_hope'],
+                                 stdout=subprocess.PIPE,
+                                 stderr=subprocess.PIPE)
+            # Windows raises IOError
+            except (IOError, OSError), err:
+                if err.errno != 2:  # ignore "no such file"
+                    raise
+
     #
     # POSIX tests
     #