]> granicus.if.org Git - python/commitdiff
Issue #15756: subprocess.poll() now properly handles errno.ECHILD to
authorGregory P. Smith <greg@krypto.org>
Sat, 29 Sep 2012 19:02:48 +0000 (12:02 -0700)
committerGregory P. Smith <greg@krypto.org>
Sat, 29 Sep 2012 19:02:48 +0000 (12:02 -0700)
return a returncode of 0 when the child has already exited or cannot
be waited on.

Lib/subprocess.py
Lib/test/subprocessdata/sigchild_ignore.py
Misc/NEWS

index 6cc3fa131286c482b88f049c0931a64ed627db62..0f284d1f0d9b5595c2e7fd87bf1309bc1b95c8b0 100644 (file)
@@ -1287,9 +1287,16 @@ class Popen(object):
                     pid, sts = _waitpid(self.pid, _WNOHANG)
                     if pid == self.pid:
                         self._handle_exitstatus(sts)
-                except _os_error:
+                except _os_error as e:
                     if _deadstate is not None:
                         self.returncode = _deadstate
+                    if e.errno == errno.ECHILD:
+                        # This happens if SIGCLD is set to be ignored or
+                        # waiting for child processes has otherwise been
+                        # disabled for our process.  This child is dead, we
+                        # can't get the status.
+                        # http://bugs.python.org/issue15756
+                        self.returncode = 0
             return self.returncode
 
 
index 6072aece28a89001e37987af2d539dff0c5498b9..86320fb35e719ffc9720b86d9a92ceccf9d64648 100644 (file)
@@ -1,6 +1,15 @@
-import signal, subprocess, sys
+import signal, subprocess, sys, time
 # On Linux this causes os.waitpid to fail with OSError as the OS has already
 # reaped our child process.  The wait() passing the OSError on to the caller
 # and causing us to exit with an error is what we are testing against.
 signal.signal(signal.SIGCHLD, signal.SIG_IGN)
 subprocess.Popen([sys.executable, '-c', 'print("albatross")']).wait()
+# Also ensure poll() handles an errno.ECHILD appropriately.
+p = subprocess.Popen([sys.executable, '-c', 'print("albatross")'])
+num_polls = 0
+while p.poll() is None:
+    # Waiting for the process to finish.
+    time.sleep(0.01)  # Avoid being a CPU busy loop.
+    num_polls += 1
+    if num_polls > 3000:
+        raise RuntimeError('poll should have returned 0 within 30 seconds')
index d6aa34b7369562fe1a18ac2dd99d3c0e742c5179..e46b9a5774af38b3c0ada1e76c66c98791c92955 100644 (file)
--- a/Misc/NEWS
+++ b/Misc/NEWS
@@ -113,6 +113,10 @@ Core and Builtins
 Library
 -------
 
+- Issue #15756: subprocess.poll() now properly handles errno.ECHILD to
+  return a returncode of 0 when the child has already exited or cannot
+  be waited on.
+
 - Issue #12376: Pass on parameters in TextTestResult.__init__ super call
 
 - Issue #15222: Insert blank line after each message in mbox mailboxes