[3.13] gh-74389: gh-70560: subprocess.Popen.communicate() now ignores stdin.flush error when closed (GH-142061) (#142065)

gh-74389: gh-70560: subprocess.Popen.communicate() now ignores stdin.flush error when closed (GH-142061)

gh-70560: gh-74389: subprocess.Popen.communicate() now ignores stdin.flush error when closed

with a unittest and news entry.
(cherry picked from commit 923056b2d4)

Co-authored-by: Gregory P. Smith <68491+gpshead@users.noreply.github.com>
This commit is contained in:
Miss Islington (bot) 2025-11-29 08:33:47 +01:00 committed by GitHub
parent 385688d4b4
commit 99b70fe4eb
No known key found for this signature in database
GPG key ID: B5690EEEBB952194
3 changed files with 20 additions and 0 deletions

View file

@ -2099,6 +2099,10 @@ def _communicate(self, input, endtime, orig_timeout):
self.stdin.flush() self.stdin.flush()
except BrokenPipeError: except BrokenPipeError:
pass # communicate() must ignore BrokenPipeError. pass # communicate() must ignore BrokenPipeError.
except ValueError:
# ignore ValueError: I/O operation on closed file.
if not self.stdin.closed:
raise
if not input: if not input:
try: try:
self.stdin.close() self.stdin.close()

View file

@ -1159,6 +1159,19 @@ def test_writes_before_communicate(self):
self.assertEqual(stdout, b"bananasplit") self.assertEqual(stdout, b"bananasplit")
self.assertEqual(stderr, b"") self.assertEqual(stderr, b"")
def test_communicate_stdin_closed_before_call(self):
# gh-70560, gh-74389: stdin.close() before communicate()
# should not raise ValueError from stdin.flush()
with subprocess.Popen([sys.executable, "-c",
'import sys; sys.exit(0)'],
stdin=subprocess.PIPE,
stdout=subprocess.PIPE,
stderr=subprocess.PIPE) as p:
p.stdin.close() # Close stdin before communicate
# This should not raise ValueError
(stdout, stderr) = p.communicate()
self.assertEqual(p.returncode, 0)
def test_universal_newlines_and_text(self): def test_universal_newlines_and_text(self):
args = [ args = [
sys.executable, "-c", sys.executable, "-c",

View file

@ -0,0 +1,3 @@
When the stdin being used by a :class:`subprocess.Popen` instance is closed,
this is now ignored in :meth:`subprocess.Popen.communicate` instead of
leaving the class in an inconsistent state.