blob: f1b26087cfd233e2a43117d04ffe2420a5e33fa6 [file] [log] [blame]
Simon Glassa2460ad2012-12-15 10:42:03 +00001# Copyright (c) 2012 The Chromium OS Authors.
2# Use of this source code is governed by a BSD-style license that can be
3# found in the LICENSE file.
4#
5# Copyright (c) 2003-2005 by Peter Astrand <astrand@lysator.liu.se>
6# Licensed to PSF under a Contributor Agreement.
7# See http://www.python.org/2.4/license for licensing details.
8
Anatolij Gustschinf2bcb322019-10-27 17:55:04 +01009"""Subprocess execution
Simon Glassa2460ad2012-12-15 10:42:03 +000010
11This module holds a subclass of subprocess.Popen with our own required
12features, mainly that we get access to the subprocess output while it
Anatolij Gustschinf2bcb322019-10-27 17:55:04 +010013is running rather than just at the end. This makes it easier to show
Simon Glassa2460ad2012-12-15 10:42:03 +000014progress information and filter output in real time.
15"""
16
17import errno
18import os
19import pty
20import select
21import subprocess
22import sys
23import unittest
24
25
26# Import these here so the caller does not need to import subprocess also.
27PIPE = subprocess.PIPE
28STDOUT = subprocess.STDOUT
29PIPE_PTY = -3 # Pipe output through a pty
30stay_alive = True
31
32
33class Popen(subprocess.Popen):
34 """Like subprocess.Popen with ptys and incremental output
35
36 This class deals with running a child process and filtering its output on
37 both stdout and stderr while it is running. We do this so we can monitor
38 progress, and possibly relay the output to the user if requested.
39
40 The class is similar to subprocess.Popen, the equivalent is something like:
41
42 Popen(args, stdout=subprocess.PIPE, stderr=subprocess.PIPE)
43
44 But this class has many fewer features, and two enhancement:
45
46 1. Rather than getting the output data only at the end, this class sends it
47 to a provided operation as it arrives.
48 2. We use pseudo terminals so that the child will hopefully flush its output
49 to us as soon as it is produced, rather than waiting for the end of a
50 line.
51
Simon Glass4c0557b2022-01-29 14:14:08 -070052 Use communicate_filter() to handle output from the subprocess.
Simon Glassa2460ad2012-12-15 10:42:03 +000053
54 """
55
56 def __init__(self, args, stdin=None, stdout=PIPE_PTY, stderr=PIPE_PTY,
Simon Glass5feeacf2019-08-24 07:22:41 -060057 shell=False, cwd=None, env=None, **kwargs):
Simon Glassa2460ad2012-12-15 10:42:03 +000058 """Cut-down constructor
59
60 Args:
61 args: Program and arguments for subprocess to execute.
62 stdin: See subprocess.Popen()
63 stdout: See subprocess.Popen(), except that we support the sentinel
64 value of cros_subprocess.PIPE_PTY.
65 stderr: See subprocess.Popen(), except that we support the sentinel
66 value of cros_subprocess.PIPE_PTY.
67 shell: See subprocess.Popen()
68 cwd: Working directory to change to for subprocess, or None if none.
69 env: Environment to use for this subprocess, or None to inherit parent.
70 kwargs: No other arguments are supported at the moment. Passing other
71 arguments will cause a ValueError to be raised.
72 """
73 stdout_pty = None
74 stderr_pty = None
75
76 if stdout == PIPE_PTY:
77 stdout_pty = pty.openpty()
78 stdout = os.fdopen(stdout_pty[1])
79 if stderr == PIPE_PTY:
80 stderr_pty = pty.openpty()
81 stderr = os.fdopen(stderr_pty[1])
82
83 super(Popen, self).__init__(args, stdin=stdin,
84 stdout=stdout, stderr=stderr, shell=shell, cwd=cwd, env=env,
85 **kwargs)
86
87 # If we're on a PTY, we passed the slave half of the PTY to the subprocess.
88 # We want to use the master half on our end from now on. Setting this here
89 # does make some assumptions about the implementation of subprocess, but
90 # those assumptions are pretty minor.
91
92 # Note that if stderr is STDOUT, then self.stderr will be set to None by
93 # this constructor.
94 if stdout_pty is not None:
95 self.stdout = os.fdopen(stdout_pty[0])
96 if stderr_pty is not None:
97 self.stderr = os.fdopen(stderr_pty[0])
98
99 # Insist that unit tests exist for other arguments we don't support.
100 if kwargs:
101 raise ValueError("Unit tests do not test extra args - please add tests")
102
Simon Glass4c0557b2022-01-29 14:14:08 -0700103 def convert_data(self, data):
Simon Glass390455c2019-05-11 12:46:39 -0600104 """Convert stdout/stderr data to the correct format for output
105
106 Args:
107 data: Data to convert, or None for ''
108
109 Returns:
110 Converted data, as bytes
111 """
112 if data is None:
113 return b''
114 return data
115
Simon Glass4c0557b2022-01-29 14:14:08 -0700116 def communicate_filter(self, output):
Simon Glassa2460ad2012-12-15 10:42:03 +0000117 """Interact with process: Read data from stdout and stderr.
118
119 This method runs until end-of-file is reached, then waits for the
120 subprocess to terminate.
121
122 The output function is sent all output from the subprocess and must be
123 defined like this:
124
Simon Glass4c0557b2022-01-29 14:14:08 -0700125 def output([self,] stream, data)
Simon Glassa2460ad2012-12-15 10:42:03 +0000126 Args:
127 stream: the stream the output was received on, which will be
128 sys.stdout or sys.stderr.
129 data: a string containing the data
130
Simon Glass146b6022021-10-19 21:43:24 -0600131 Returns:
132 True to terminate the process
133
Simon Glassa2460ad2012-12-15 10:42:03 +0000134 Note: The data read is buffered in memory, so do not use this
135 method if the data size is large or unlimited.
136
137 Args:
138 output: Function to call with each fragment of output.
139
140 Returns:
141 A tuple (stdout, stderr, combined) which is the data received on
142 stdout, stderr and the combined data (interleaved stdout and stderr).
143
144 Note that the interleaved output will only be sensible if you have
145 set both stdout and stderr to PIPE or PIPE_PTY. Even then it depends on
146 the timing of the output in the subprocess. If a subprocess flips
147 between stdout and stderr quickly in succession, by the time we come to
148 read the output from each we may see several lines in each, and will read
149 all the stdout lines, then all the stderr lines. So the interleaving
150 may not be correct. In this case you might want to pass
151 stderr=cros_subprocess.STDOUT to the constructor.
152
153 This feature is still useful for subprocesses where stderr is
154 rarely used and indicates an error.
155
156 Note also that if you set stderr to STDOUT, then stderr will be empty
157 and the combined output will just be the same as stdout.
158 """
159
160 read_set = []
161 write_set = []
162 stdout = None # Return
163 stderr = None # Return
164
165 if self.stdin:
166 # Flush stdio buffer. This might block, if the user has
167 # been writing to .stdin in an uncontrolled fashion.
168 self.stdin.flush()
169 if input:
170 write_set.append(self.stdin)
171 else:
172 self.stdin.close()
173 if self.stdout:
174 read_set.append(self.stdout)
Simon Glass1e689d22021-07-06 10:36:40 -0600175 stdout = bytearray()
Simon Glassa2460ad2012-12-15 10:42:03 +0000176 if self.stderr and self.stderr != self.stdout:
177 read_set.append(self.stderr)
Simon Glass1e689d22021-07-06 10:36:40 -0600178 stderr = bytearray()
179 combined = bytearray()
Simon Glassa2460ad2012-12-15 10:42:03 +0000180
Simon Glass146b6022021-10-19 21:43:24 -0600181 stop_now = False
Simon Glassa2460ad2012-12-15 10:42:03 +0000182 input_offset = 0
183 while read_set or write_set:
184 try:
185 rlist, wlist, _ = select.select(read_set, write_set, [], 0.2)
Paul Burtonf14a1312016-09-27 16:03:51 +0100186 except select.error as e:
Simon Glassa2460ad2012-12-15 10:42:03 +0000187 if e.args[0] == errno.EINTR:
188 continue
189 raise
190
191 if not stay_alive:
192 self.terminate()
193
194 if self.stdin in wlist:
195 # When select has indicated that the file is writable,
196 # we can write up to PIPE_BUF bytes without risk
197 # blocking. POSIX defines PIPE_BUF >= 512
198 chunk = input[input_offset : input_offset + 512]
199 bytes_written = os.write(self.stdin.fileno(), chunk)
200 input_offset += bytes_written
201 if input_offset >= len(input):
202 self.stdin.close()
203 write_set.remove(self.stdin)
204
205 if self.stdout in rlist:
Simon Glass390455c2019-05-11 12:46:39 -0600206 data = b''
Simon Glassa2460ad2012-12-15 10:42:03 +0000207 # We will get an error on read if the pty is closed
208 try:
209 data = os.read(self.stdout.fileno(), 1024)
210 except OSError:
211 pass
Simon Glass390455c2019-05-11 12:46:39 -0600212 if not len(data):
Simon Glassa2460ad2012-12-15 10:42:03 +0000213 self.stdout.close()
214 read_set.remove(self.stdout)
215 else:
Simon Glass390455c2019-05-11 12:46:39 -0600216 stdout += data
217 combined += data
Simon Glassa2460ad2012-12-15 10:42:03 +0000218 if output:
Simon Glass146b6022021-10-19 21:43:24 -0600219 stop_now = output(sys.stdout, data)
Simon Glassa2460ad2012-12-15 10:42:03 +0000220 if self.stderr in rlist:
Simon Glass390455c2019-05-11 12:46:39 -0600221 data = b''
Simon Glassa2460ad2012-12-15 10:42:03 +0000222 # We will get an error on read if the pty is closed
223 try:
224 data = os.read(self.stderr.fileno(), 1024)
Simon Glassa2460ad2012-12-15 10:42:03 +0000225 except OSError:
226 pass
Simon Glass390455c2019-05-11 12:46:39 -0600227 if not len(data):
Simon Glassa2460ad2012-12-15 10:42:03 +0000228 self.stderr.close()
229 read_set.remove(self.stderr)
230 else:
Simon Glass390455c2019-05-11 12:46:39 -0600231 stderr += data
232 combined += data
Simon Glassa2460ad2012-12-15 10:42:03 +0000233 if output:
Simon Glass146b6022021-10-19 21:43:24 -0600234 stop_now = output(sys.stderr, data)
235 if stop_now:
236 self.terminate()
Simon Glassa2460ad2012-12-15 10:42:03 +0000237
238 # All data exchanged. Translate lists into strings.
Simon Glass4c0557b2022-01-29 14:14:08 -0700239 stdout = self.convert_data(stdout)
240 stderr = self.convert_data(stderr)
241 combined = self.convert_data(combined)
Simon Glassa2460ad2012-12-15 10:42:03 +0000242
243 # Translate newlines, if requested. We cannot let the file
244 # object do the translation: It is based on stdio, which is
245 # impossible to combine with select (unless forcing no
246 # buffering).
247 if self.universal_newlines and hasattr(file, 'newlines'):
248 if stdout:
249 stdout = self._translate_newlines(stdout)
250 if stderr:
251 stderr = self._translate_newlines(stderr)
252
253 self.wait()
254 return (stdout, stderr, combined)
255
256
257# Just being a unittest.TestCase gives us 14 public methods. Unless we
258# disable this, we can only have 6 tests in a TestCase. That's not enough.
259#
260# pylint: disable=R0904
261
262class TestSubprocess(unittest.TestCase):
263 """Our simple unit test for this module"""
264
265 class MyOperation:
266 """Provides a operation that we can pass to Popen"""
267 def __init__(self, input_to_send=None):
268 """Constructor to set up the operation and possible input.
269
270 Args:
271 input_to_send: a text string to send when we first get input. We will
272 add \r\n to the string.
273 """
274 self.stdout_data = ''
275 self.stderr_data = ''
276 self.combined_data = ''
277 self.stdin_pipe = None
278 self._input_to_send = input_to_send
279 if input_to_send:
280 pipe = os.pipe()
281 self.stdin_read_pipe = pipe[0]
282 self._stdin_write_pipe = os.fdopen(pipe[1], 'w')
283
Simon Glass4c0557b2022-01-29 14:14:08 -0700284 def output(self, stream, data):
Simon Glassa2460ad2012-12-15 10:42:03 +0000285 """Output handler for Popen. Stores the data for later comparison"""
286 if stream == sys.stdout:
287 self.stdout_data += data
288 if stream == sys.stderr:
289 self.stderr_data += data
290 self.combined_data += data
291
292 # Output the input string if we have one.
293 if self._input_to_send:
294 self._stdin_write_pipe.write(self._input_to_send + '\r\n')
295 self._stdin_write_pipe.flush()
296
Simon Glass4c0557b2022-01-29 14:14:08 -0700297 def _basic_check(self, plist, oper):
Simon Glassa2460ad2012-12-15 10:42:03 +0000298 """Basic checks that the output looks sane."""
299 self.assertEqual(plist[0], oper.stdout_data)
300 self.assertEqual(plist[1], oper.stderr_data)
301 self.assertEqual(plist[2], oper.combined_data)
302
303 # The total length of stdout and stderr should equal the combined length
304 self.assertEqual(len(plist[0]) + len(plist[1]), len(plist[2]))
305
306 def test_simple(self):
307 """Simple redirection: Get process list"""
308 oper = TestSubprocess.MyOperation()
Simon Glass4c0557b2022-01-29 14:14:08 -0700309 plist = Popen(['ps']).communicate_filter(oper.output)
310 self._basic_check(plist, oper)
Simon Glassa2460ad2012-12-15 10:42:03 +0000311
312 def test_stderr(self):
313 """Check stdout and stderr"""
314 oper = TestSubprocess.MyOperation()
315 cmd = 'echo fred >/dev/stderr && false || echo bad'
Simon Glass4c0557b2022-01-29 14:14:08 -0700316 plist = Popen([cmd], shell=True).communicate_filter(oper.output)
317 self._basic_check(plist, oper)
Simon Glassa2460ad2012-12-15 10:42:03 +0000318 self.assertEqual(plist [0], 'bad\r\n')
319 self.assertEqual(plist [1], 'fred\r\n')
320
321 def test_shell(self):
322 """Check with and without shell works"""
323 oper = TestSubprocess.MyOperation()
324 cmd = 'echo test >/dev/stderr'
325 self.assertRaises(OSError, Popen, [cmd], shell=False)
Simon Glass4c0557b2022-01-29 14:14:08 -0700326 plist = Popen([cmd], shell=True).communicate_filter(oper.output)
327 self._basic_check(plist, oper)
Simon Glassa2460ad2012-12-15 10:42:03 +0000328 self.assertEqual(len(plist [0]), 0)
329 self.assertEqual(plist [1], 'test\r\n')
330
331 def test_list_args(self):
332 """Check with and without shell works using list arguments"""
333 oper = TestSubprocess.MyOperation()
334 cmd = ['echo', 'test', '>/dev/stderr']
Simon Glass4c0557b2022-01-29 14:14:08 -0700335 plist = Popen(cmd, shell=False).communicate_filter(oper.output)
336 self._basic_check(plist, oper)
Simon Glassa2460ad2012-12-15 10:42:03 +0000337 self.assertEqual(plist [0], ' '.join(cmd[1:]) + '\r\n')
338 self.assertEqual(len(plist [1]), 0)
339
340 oper = TestSubprocess.MyOperation()
341
342 # this should be interpreted as 'echo' with the other args dropped
343 cmd = ['echo', 'test', '>/dev/stderr']
Simon Glass4c0557b2022-01-29 14:14:08 -0700344 plist = Popen(cmd, shell=True).communicate_filter(oper.output)
345 self._basic_check(plist, oper)
Simon Glassa2460ad2012-12-15 10:42:03 +0000346 self.assertEqual(plist [0], '\r\n')
347
348 def test_cwd(self):
349 """Check we can change directory"""
350 for shell in (False, True):
351 oper = TestSubprocess.MyOperation()
Simon Glass4c0557b2022-01-29 14:14:08 -0700352 plist = Popen('pwd', shell=shell, cwd='/tmp').communicate_filter(
353 oper.output)
354 self._basic_check(plist, oper)
Simon Glassa2460ad2012-12-15 10:42:03 +0000355 self.assertEqual(plist [0], '/tmp\r\n')
356
357 def test_env(self):
358 """Check we can change environment"""
359 for add in (False, True):
360 oper = TestSubprocess.MyOperation()
361 env = os.environ
362 if add:
363 env ['FRED'] = 'fred'
364 cmd = 'echo $FRED'
Simon Glass4c0557b2022-01-29 14:14:08 -0700365 plist = Popen(cmd, shell=True, env=env).communicate_filter(oper.output)
366 self._basic_check(plist, oper)
Simon Glassa2460ad2012-12-15 10:42:03 +0000367 self.assertEqual(plist [0], add and 'fred\r\n' or '\r\n')
368
369 def test_extra_args(self):
370 """Check we can't add extra arguments"""
371 self.assertRaises(ValueError, Popen, 'true', close_fds=False)
372
373 def test_basic_input(self):
374 """Check that incremental input works
375
376 We set up a subprocess which will prompt for name. When we see this prompt
377 we send the name as input to the process. It should then print the name
378 properly to stdout.
379 """
380 oper = TestSubprocess.MyOperation('Flash')
381 prompt = 'What is your name?: '
382 cmd = 'echo -n "%s"; read name; echo Hello $name' % prompt
383 plist = Popen([cmd], stdin=oper.stdin_read_pipe,
Simon Glass4c0557b2022-01-29 14:14:08 -0700384 shell=True).communicate_filter(oper.output)
385 self._basic_check(plist, oper)
Simon Glassa2460ad2012-12-15 10:42:03 +0000386 self.assertEqual(len(plist [1]), 0)
387 self.assertEqual(plist [0], prompt + 'Hello Flash\r\r\n')
388
389 def test_isatty(self):
390 """Check that ptys appear as terminals to the subprocess"""
391 oper = TestSubprocess.MyOperation()
392 cmd = ('if [ -t %d ]; then echo "terminal %d" >&%d; '
393 'else echo "not %d" >&%d; fi;')
394 both_cmds = ''
395 for fd in (1, 2):
396 both_cmds += cmd % (fd, fd, fd, fd, fd)
Simon Glass4c0557b2022-01-29 14:14:08 -0700397 plist = Popen(both_cmds, shell=True).communicate_filter(oper.output)
398 self._basic_check(plist, oper)
Simon Glassa2460ad2012-12-15 10:42:03 +0000399 self.assertEqual(plist [0], 'terminal 1\r\n')
400 self.assertEqual(plist [1], 'terminal 2\r\n')
401
402 # Now try with PIPE and make sure it is not a terminal
403 oper = TestSubprocess.MyOperation()
404 plist = Popen(both_cmds, stdout=subprocess.PIPE, stderr=subprocess.PIPE,
Simon Glass4c0557b2022-01-29 14:14:08 -0700405 shell=True).communicate_filter(oper.output)
406 self._basic_check(plist, oper)
Simon Glassa2460ad2012-12-15 10:42:03 +0000407 self.assertEqual(plist [0], 'not 1\n')
408 self.assertEqual(plist [1], 'not 2\n')
409
410if __name__ == '__main__':
411 unittest.main()