]> git.ipfire.org Git - thirdparty/Python/cpython.git/commitdiff
Issue #11459: A `bufsize` value of 0 in subprocess.Popen() really creates
authorAntoine Pitrou <solipsis@pitrou.net>
Sat, 19 Mar 2011 16:00:37 +0000 (17:00 +0100)
committerAntoine Pitrou <solipsis@pitrou.net>
Sat, 19 Mar 2011 16:00:37 +0000 (17:00 +0100)
unbuffered pipes, such that select() works properly on them.

Doc/library/platform.rst
Lib/os.py
Lib/platform.py
Lib/subprocess.py
Lib/test/test_subprocess.py
Misc/NEWS

index 56518f79e84c6790a4b4c9a9e49b5c286cec0c79..a71d5335a2cc769a5121dbe4e7ade9248d63dd75 100644 (file)
@@ -205,7 +205,7 @@ Windows Platform
 Win95/98 specific
 ^^^^^^^^^^^^^^^^^
 
-.. function:: popen(cmd, mode='r', bufsize=None)
+.. function:: popen(cmd, mode='r', bufsize=-1)
 
    Portable :func:`popen` interface.  Find a working popen implementation
    preferring :func:`win32pipe.popen`.  On Windows NT, :func:`win32pipe.popen`
index aa1a40f7a754e3613aab3503bb4ea57b1316b37b..b46c02f5800fda85e444b1e6ccb405db64010612 100644 (file)
--- a/Lib/os.py
+++ b/Lib/os.py
@@ -629,11 +629,13 @@ if not _exists("urandom"):
         return bs
 
 # Supply os.popen()
-def popen(cmd, mode="r", buffering=None):
+def popen(cmd, mode="r", buffering=-1):
     if not isinstance(cmd, str):
         raise TypeError("invalid cmd type (%s, expected string)" % type(cmd))
     if mode not in ("r", "w"):
         raise ValueError("invalid mode %r" % mode)
+    if buffering == 0 or buffering == None:
+        raise ValueError("popen() does not support unbuffered streams")
     import subprocess, io
     if mode == "r":
         proc = subprocess.Popen(cmd,
index 2294604ea41a71cbdd812c363c1f507e0ece5f39..dd09460aad5b05e59800f2c9f27a8055377e8544 100755 (executable)
@@ -398,7 +398,7 @@ class _popen:
     # Alias
     __del__ = close
 
-def popen(cmd, mode='r', bufsize=None):
+def popen(cmd, mode='r', bufsize=-1):
 
     """ Portable popen() interface.
     """
index 97792dbe605e9642e1f60262225d6442200c0748..dc5b6088da63582e2982a677882af461f7cf8169 100644 (file)
@@ -653,8 +653,6 @@ class Popen(object):
             if errread is not None:
                 errread = msvcrt.open_osfhandle(errread.Detach(), 0)
 
-        if bufsize == 0:
-            bufsize = 1  # Nearly unbuffered (XXX for now)
         if p2cwrite is not None:
             self.stdin = io.open(p2cwrite, 'wb', bufsize)
             if self.universal_newlines:
index c781904ed9c8690cdc82bd662a7ee0b71dc497d4..410849fc286ece42a48d28c1ab7b938b9205000b 100644 (file)
@@ -864,6 +864,21 @@ class ProcessTestCase(BaseTestCase):
             def prepare():
                 raise ValueError("surrogate:\uDCff")
 
+        def test_select_unbuffered(self):
+            # Issue #11459: bufsize=0 should really set the pipes as
+            # unbuffered (and therefore let select() work properly).
+            select = support.import_module("select")
+            p = subprocess.Popen([sys.executable, "-c",
+                                  'import sys;'
+                                  'sys.stdout.write("apple")'],
+                                 stdout=subprocess.PIPE,
+                                 bufsize=0)
+            f = p.stdout
+            try:
+                self.assertEqual(f.read(4), b"appl")
+                self.assertIn(f, select.select([f], [], [], 0.0)[0])
+            finally:
+                p.wait()
 
     #
     # Windows tests
index f18e557fd00627e042999eccd9399991afc3ceea..be44a06ac14bfbcd1ea2003684f4be275b8faea5 100644 (file)
--- a/Misc/NEWS
+++ b/Misc/NEWS
@@ -40,6 +40,9 @@ Core and Builtins
 Library
 -------
 
+- Issue #11459: A ``bufsize`` value of 0 in subprocess.Popen() really creates
+  unbuffered pipes, such that select() works properly on them.
+
 - Issue #5421: Fix misleading error message when one of socket.sendto()'s
   arguments has the wrong type.  Patch by Nikita Vetoshkin.