blob: 3c926755ce5c33a4dd7dced993b6d013a0ff1fdf [file] [log] [blame]
Fredrik Lundh5b3687d2004-10-12 15:26:28 +00001import unittest
2from test import test_support
3import subprocess
4import sys
5import signal
6import os
7import tempfile
8import time
Tim Peters3761e8d2004-10-13 04:07:12 +00009import re
Fredrik Lundh5b3687d2004-10-12 15:26:28 +000010
11mswindows = (sys.platform == "win32")
12
13#
14# Depends on the following external programs: Python
15#
16
17if mswindows:
Tim Peters3b01a702004-10-12 22:19:32 +000018 SETBINARY = ('import msvcrt; msvcrt.setmode(sys.stdout.fileno(), '
19 'os.O_BINARY);')
Fredrik Lundh5b3687d2004-10-12 15:26:28 +000020else:
21 SETBINARY = ''
22
Tim Peters3761e8d2004-10-13 04:07:12 +000023# In a debug build, stuff like "[6580 refs]" is printed to stderr at
24# shutdown time. That frustrates tests trying to check stderr produced
25# from a spawned Python process.
26def remove_stderr_debug_decorations(stderr):
Guido van Rossumc9e363c2007-05-15 23:18:55 +000027 return re.sub(r"\[\d+ refs\]\r?\n?$", "", str8(stderr))
Tim Peters3761e8d2004-10-13 04:07:12 +000028
Fredrik Lundh5b3687d2004-10-12 15:26:28 +000029class ProcessTestCase(unittest.TestCase):
Thomas Wouters0e3f5912006-08-11 14:57:12 +000030 def setUp(self):
31 # Try to minimize the number of children we have so this test
32 # doesn't crash on some buildbots (Alphas in particular).
33 if hasattr(test_support, "reap_children"):
34 test_support.reap_children()
35
36 def tearDown(self):
37 # Try to minimize the number of children we have so this test
38 # doesn't crash on some buildbots (Alphas in particular).
39 if hasattr(test_support, "reap_children"):
40 test_support.reap_children()
41
Fredrik Lundh5b3687d2004-10-12 15:26:28 +000042 def mkstemp(self):
43 """wrapper for mkstemp, calling mktemp if mkstemp is not available"""
44 if hasattr(tempfile, "mkstemp"):
45 return tempfile.mkstemp()
46 else:
47 fname = tempfile.mktemp()
48 return os.open(fname, os.O_RDWR|os.O_CREAT), fname
Tim Peterse718f612004-10-12 21:51:32 +000049
Fredrik Lundh5b3687d2004-10-12 15:26:28 +000050 #
51 # Generic tests
52 #
53 def test_call_seq(self):
Tim Peters7b759da2004-10-12 22:29:54 +000054 # call() function with sequence argument
Tim Peters3b01a702004-10-12 22:19:32 +000055 rc = subprocess.call([sys.executable, "-c",
56 "import sys; sys.exit(47)"])
Fredrik Lundh5b3687d2004-10-12 15:26:28 +000057 self.assertEqual(rc, 47)
58
Peter Astrand454f7672005-01-01 09:36:35 +000059 def test_check_call_zero(self):
60 # check_call() function with zero return code
61 rc = subprocess.check_call([sys.executable, "-c",
62 "import sys; sys.exit(0)"])
63 self.assertEqual(rc, 0)
64
65 def test_check_call_nonzero(self):
66 # check_call() function with non-zero return code
67 try:
68 subprocess.check_call([sys.executable, "-c",
69 "import sys; sys.exit(47)"])
Guido van Rossumb940e112007-01-10 16:19:56 +000070 except subprocess.CalledProcessError as e:
Thomas Wouters0e3f5912006-08-11 14:57:12 +000071 self.assertEqual(e.returncode, 47)
Peter Astrand454f7672005-01-01 09:36:35 +000072 else:
73 self.fail("Expected CalledProcessError")
74
Fredrik Lundh5b3687d2004-10-12 15:26:28 +000075 def test_call_kwargs(self):
Tim Peters7b759da2004-10-12 22:29:54 +000076 # call() function with keyword args
Fredrik Lundh5b3687d2004-10-12 15:26:28 +000077 newenv = os.environ.copy()
78 newenv["FRUIT"] = "banana"
79 rc = subprocess.call([sys.executable, "-c",
80 'import sys, os;' \
81 'sys.exit(os.getenv("FRUIT")=="banana")'],
82 env=newenv)
83 self.assertEqual(rc, 1)
84
85 def test_stdin_none(self):
Tim Peters7b759da2004-10-12 22:29:54 +000086 # .stdin is None when not redirected
Georg Brandl88fc6642007-02-09 21:28:07 +000087 p = subprocess.Popen([sys.executable, "-c", 'print("banana")'],
Fredrik Lundh5b3687d2004-10-12 15:26:28 +000088 stdout=subprocess.PIPE, stderr=subprocess.PIPE)
89 p.wait()
90 self.assertEqual(p.stdin, None)
91
92 def test_stdout_none(self):
Tim Peters7b759da2004-10-12 22:29:54 +000093 # .stdout is None when not redirected
Tim Peters29b6b4f2004-10-13 03:43:40 +000094 p = subprocess.Popen([sys.executable, "-c",
Georg Brandl88fc6642007-02-09 21:28:07 +000095 'print(" this bit of output is from a '
Tim Peters4052fe52004-10-13 03:29:54 +000096 'test of stdout in a different '
Georg Brandl88fc6642007-02-09 21:28:07 +000097 'process ...")'],
Tim Peters4052fe52004-10-13 03:29:54 +000098 stdin=subprocess.PIPE, stderr=subprocess.PIPE)
Fredrik Lundh5b3687d2004-10-12 15:26:28 +000099 p.wait()
100 self.assertEqual(p.stdout, None)
101
102 def test_stderr_none(self):
Tim Peters7b759da2004-10-12 22:29:54 +0000103 # .stderr is None when not redirected
Georg Brandl88fc6642007-02-09 21:28:07 +0000104 p = subprocess.Popen([sys.executable, "-c", 'print("banana")'],
Fredrik Lundh5b3687d2004-10-12 15:26:28 +0000105 stdin=subprocess.PIPE, stdout=subprocess.PIPE)
106 p.wait()
107 self.assertEqual(p.stderr, None)
108
109 def test_executable(self):
Tim Peters3b01a702004-10-12 22:19:32 +0000110 p = subprocess.Popen(["somethingyoudonthave",
111 "-c", "import sys; sys.exit(47)"],
112 executable=sys.executable)
Fredrik Lundh5b3687d2004-10-12 15:26:28 +0000113 p.wait()
114 self.assertEqual(p.returncode, 47)
115
116 def test_stdin_pipe(self):
Tim Peters7b759da2004-10-12 22:29:54 +0000117 # stdin redirection
Fredrik Lundh5b3687d2004-10-12 15:26:28 +0000118 p = subprocess.Popen([sys.executable, "-c",
119 'import sys; sys.exit(sys.stdin.read() == "pear")'],
120 stdin=subprocess.PIPE)
Guido van Rossumbb839ef2007-08-27 23:58:21 +0000121 p.stdin.write(b"pear")
Fredrik Lundh5b3687d2004-10-12 15:26:28 +0000122 p.stdin.close()
123 p.wait()
124 self.assertEqual(p.returncode, 1)
125
126 def test_stdin_filedes(self):
Tim Peters7b759da2004-10-12 22:29:54 +0000127 # stdin is set to open file descriptor
Tim Peterse718f612004-10-12 21:51:32 +0000128 tf = tempfile.TemporaryFile()
Fredrik Lundh5b3687d2004-10-12 15:26:28 +0000129 d = tf.fileno()
130 os.write(d, "pear")
131 os.lseek(d, 0, 0)
132 p = subprocess.Popen([sys.executable, "-c",
133 'import sys; sys.exit(sys.stdin.read() == "pear")'],
134 stdin=d)
135 p.wait()
136 self.assertEqual(p.returncode, 1)
137
138 def test_stdin_fileobj(self):
Tim Peters7b759da2004-10-12 22:29:54 +0000139 # stdin is set to open file object
Fredrik Lundh5b3687d2004-10-12 15:26:28 +0000140 tf = tempfile.TemporaryFile()
Guido van Rossumbb839ef2007-08-27 23:58:21 +0000141 tf.write(b"pear")
Fredrik Lundh5b3687d2004-10-12 15:26:28 +0000142 tf.seek(0)
143 p = subprocess.Popen([sys.executable, "-c",
144 'import sys; sys.exit(sys.stdin.read() == "pear")'],
145 stdin=tf)
146 p.wait()
147 self.assertEqual(p.returncode, 1)
148
149 def test_stdout_pipe(self):
Tim Peters7b759da2004-10-12 22:29:54 +0000150 # stdout redirection
Fredrik Lundh5b3687d2004-10-12 15:26:28 +0000151 p = subprocess.Popen([sys.executable, "-c",
152 'import sys; sys.stdout.write("orange")'],
153 stdout=subprocess.PIPE)
Guido van Rossumfa0054a2007-05-24 04:05:35 +0000154 self.assertEqual(p.stdout.read(), b"orange")
Fredrik Lundh5b3687d2004-10-12 15:26:28 +0000155
156 def test_stdout_filedes(self):
Tim Peters7b759da2004-10-12 22:29:54 +0000157 # stdout is set to open file descriptor
Tim Peterse718f612004-10-12 21:51:32 +0000158 tf = tempfile.TemporaryFile()
Fredrik Lundh5b3687d2004-10-12 15:26:28 +0000159 d = tf.fileno()
160 p = subprocess.Popen([sys.executable, "-c",
161 'import sys; sys.stdout.write("orange")'],
162 stdout=d)
163 p.wait()
164 os.lseek(d, 0, 0)
Guido van Rossumc9e363c2007-05-15 23:18:55 +0000165 self.assertEqual(os.read(d, 1024), b"orange")
Fredrik Lundh5b3687d2004-10-12 15:26:28 +0000166
167 def test_stdout_fileobj(self):
Tim Peters7b759da2004-10-12 22:29:54 +0000168 # stdout is set to open file object
Tim Peterse718f612004-10-12 21:51:32 +0000169 tf = tempfile.TemporaryFile()
Fredrik Lundh5b3687d2004-10-12 15:26:28 +0000170 p = subprocess.Popen([sys.executable, "-c",
171 'import sys; sys.stdout.write("orange")'],
172 stdout=tf)
173 p.wait()
174 tf.seek(0)
Guido van Rossumfa0054a2007-05-24 04:05:35 +0000175 self.assertEqual(tf.read(), b"orange")
Fredrik Lundh5b3687d2004-10-12 15:26:28 +0000176
177 def test_stderr_pipe(self):
Tim Peters7b759da2004-10-12 22:29:54 +0000178 # stderr redirection
Fredrik Lundh5b3687d2004-10-12 15:26:28 +0000179 p = subprocess.Popen([sys.executable, "-c",
180 'import sys; sys.stderr.write("strawberry")'],
181 stderr=subprocess.PIPE)
Tim Peters3761e8d2004-10-13 04:07:12 +0000182 self.assertEqual(remove_stderr_debug_decorations(p.stderr.read()),
183 "strawberry")
Fredrik Lundh5b3687d2004-10-12 15:26:28 +0000184
185 def test_stderr_filedes(self):
Tim Peters7b759da2004-10-12 22:29:54 +0000186 # stderr is set to open file descriptor
Tim Peterse718f612004-10-12 21:51:32 +0000187 tf = tempfile.TemporaryFile()
Fredrik Lundh5b3687d2004-10-12 15:26:28 +0000188 d = tf.fileno()
189 p = subprocess.Popen([sys.executable, "-c",
190 'import sys; sys.stderr.write("strawberry")'],
191 stderr=d)
192 p.wait()
193 os.lseek(d, 0, 0)
Tim Peters3761e8d2004-10-13 04:07:12 +0000194 self.assertEqual(remove_stderr_debug_decorations(os.read(d, 1024)),
195 "strawberry")
Fredrik Lundh5b3687d2004-10-12 15:26:28 +0000196
197 def test_stderr_fileobj(self):
Tim Peters7b759da2004-10-12 22:29:54 +0000198 # stderr is set to open file object
Tim Peterse718f612004-10-12 21:51:32 +0000199 tf = tempfile.TemporaryFile()
Fredrik Lundh5b3687d2004-10-12 15:26:28 +0000200 p = subprocess.Popen([sys.executable, "-c",
201 'import sys; sys.stderr.write("strawberry")'],
202 stderr=tf)
203 p.wait()
204 tf.seek(0)
Tim Peters3761e8d2004-10-13 04:07:12 +0000205 self.assertEqual(remove_stderr_debug_decorations(tf.read()),
206 "strawberry")
Fredrik Lundh5b3687d2004-10-12 15:26:28 +0000207
208 def test_stdout_stderr_pipe(self):
Tim Peters7b759da2004-10-12 22:29:54 +0000209 # capture stdout and stderr to the same pipe
Fredrik Lundh5b3687d2004-10-12 15:26:28 +0000210 p = subprocess.Popen([sys.executable, "-c",
211 'import sys;' \
212 'sys.stdout.write("apple");' \
213 'sys.stdout.flush();' \
214 'sys.stderr.write("orange")'],
215 stdout=subprocess.PIPE,
216 stderr=subprocess.STDOUT)
Tim Peters3761e8d2004-10-13 04:07:12 +0000217 output = p.stdout.read()
218 stripped = remove_stderr_debug_decorations(output)
219 self.assertEqual(stripped, "appleorange")
Fredrik Lundh5b3687d2004-10-12 15:26:28 +0000220
221 def test_stdout_stderr_file(self):
Tim Peters7b759da2004-10-12 22:29:54 +0000222 # capture stdout and stderr to the same open file
Fredrik Lundh5b3687d2004-10-12 15:26:28 +0000223 tf = tempfile.TemporaryFile()
224 p = subprocess.Popen([sys.executable, "-c",
225 'import sys;' \
226 'sys.stdout.write("apple");' \
227 'sys.stdout.flush();' \
228 'sys.stderr.write("orange")'],
229 stdout=tf,
230 stderr=tf)
231 p.wait()
232 tf.seek(0)
Tim Peters3761e8d2004-10-13 04:07:12 +0000233 output = tf.read()
234 stripped = remove_stderr_debug_decorations(output)
235 self.assertEqual(stripped, "appleorange")
Fredrik Lundh5b3687d2004-10-12 15:26:28 +0000236
Thomas Wouters89f507f2006-12-13 04:49:30 +0000237 def test_stdout_filedes_of_stdout(self):
238 # stdout is set to 1 (#1531862).
239 cmd = r"import sys, os; sys.exit(os.write(sys.stdout.fileno(), '.\n'))"
240 rc = subprocess.call([sys.executable, "-c", cmd], stdout=1)
241 self.assertEquals(rc, 2)
242
Fredrik Lundh5b3687d2004-10-12 15:26:28 +0000243 def test_cwd(self):
Fredrik Lundh5b3687d2004-10-12 15:26:28 +0000244 tmpdir = os.getenv("TEMP", "/tmp")
Peter Astrand195404f2004-11-12 15:51:48 +0000245 # We cannot use os.path.realpath to canonicalize the path,
246 # since it doesn't expand Tru64 {memb} strings. See bug 1063571.
247 cwd = os.getcwd()
248 os.chdir(tmpdir)
249 tmpdir = os.getcwd()
250 os.chdir(cwd)
Fredrik Lundh5b3687d2004-10-12 15:26:28 +0000251 p = subprocess.Popen([sys.executable, "-c",
252 'import sys,os;' \
253 'sys.stdout.write(os.getcwd())'],
254 stdout=subprocess.PIPE,
255 cwd=tmpdir)
Fredrik Lundh59c05592004-10-13 06:55:40 +0000256 normcase = os.path.normcase
Guido van Rossumbb839ef2007-08-27 23:58:21 +0000257 self.assertEqual(normcase(p.stdout.read().decode("utf-8")),
258 normcase(tmpdir))
Fredrik Lundh5b3687d2004-10-12 15:26:28 +0000259
260 def test_env(self):
Fredrik Lundh5b3687d2004-10-12 15:26:28 +0000261 newenv = os.environ.copy()
262 newenv["FRUIT"] = "orange"
263 p = subprocess.Popen([sys.executable, "-c",
264 'import sys,os;' \
265 'sys.stdout.write(os.getenv("FRUIT"))'],
266 stdout=subprocess.PIPE,
267 env=newenv)
Guido van Rossumfa0054a2007-05-24 04:05:35 +0000268 self.assertEqual(p.stdout.read(), b"orange")
Fredrik Lundh5b3687d2004-10-12 15:26:28 +0000269
Peter Astrandcbac93c2005-03-03 20:24:28 +0000270 def test_communicate_stdin(self):
271 p = subprocess.Popen([sys.executable, "-c",
272 'import sys; sys.exit(sys.stdin.read() == "pear")'],
273 stdin=subprocess.PIPE)
Guido van Rossumbb839ef2007-08-27 23:58:21 +0000274 p.communicate(b"pear")
Peter Astrandcbac93c2005-03-03 20:24:28 +0000275 self.assertEqual(p.returncode, 1)
276
277 def test_communicate_stdout(self):
278 p = subprocess.Popen([sys.executable, "-c",
279 'import sys; sys.stdout.write("pineapple")'],
280 stdout=subprocess.PIPE)
281 (stdout, stderr) = p.communicate()
Guido van Rossumfa0054a2007-05-24 04:05:35 +0000282 self.assertEqual(stdout, b"pineapple")
Peter Astrandcbac93c2005-03-03 20:24:28 +0000283 self.assertEqual(stderr, None)
284
285 def test_communicate_stderr(self):
286 p = subprocess.Popen([sys.executable, "-c",
287 'import sys; sys.stderr.write("pineapple")'],
288 stderr=subprocess.PIPE)
289 (stdout, stderr) = p.communicate()
290 self.assertEqual(stdout, None)
Brett Cannon653a5ad2005-03-05 06:40:52 +0000291 # When running with a pydebug build, the # of references is outputted
292 # to stderr, so just check if stderr at least started with "pinapple"
Guido van Rossumbb839ef2007-08-27 23:58:21 +0000293 self.assert_(stderr.startswith(b"pineapple"))
Peter Astrandcbac93c2005-03-03 20:24:28 +0000294
Fredrik Lundh5b3687d2004-10-12 15:26:28 +0000295 def test_communicate(self):
Fredrik Lundh5b3687d2004-10-12 15:26:28 +0000296 p = subprocess.Popen([sys.executable, "-c",
297 'import sys,os;' \
298 'sys.stderr.write("pineapple");' \
299 'sys.stdout.write(sys.stdin.read())'],
Tim Peters3b01a702004-10-12 22:19:32 +0000300 stdin=subprocess.PIPE,
301 stdout=subprocess.PIPE,
302 stderr=subprocess.PIPE)
Fredrik Lundh5b3687d2004-10-12 15:26:28 +0000303 (stdout, stderr) = p.communicate("banana")
Guido van Rossumc9e363c2007-05-15 23:18:55 +0000304 self.assertEqual(stdout, b"banana")
Tim Peters3761e8d2004-10-13 04:07:12 +0000305 self.assertEqual(remove_stderr_debug_decorations(stderr),
306 "pineapple")
Fredrik Lundh5b3687d2004-10-12 15:26:28 +0000307
308 def test_communicate_returns(self):
Tim Peters7b759da2004-10-12 22:29:54 +0000309 # communicate() should return None if no redirection is active
Tim Peters3b01a702004-10-12 22:19:32 +0000310 p = subprocess.Popen([sys.executable, "-c",
311 "import sys; sys.exit(47)"])
Fredrik Lundh5b3687d2004-10-12 15:26:28 +0000312 (stdout, stderr) = p.communicate()
313 self.assertEqual(stdout, None)
314 self.assertEqual(stderr, None)
315
316 def test_communicate_pipe_buf(self):
Tim Peters7b759da2004-10-12 22:29:54 +0000317 # communicate() with writes larger than pipe_buf
Fredrik Lundh5b3687d2004-10-12 15:26:28 +0000318 # This test will probably deadlock rather than fail, if
Tim Peterse718f612004-10-12 21:51:32 +0000319 # communicate() does not work properly.
Fredrik Lundh5b3687d2004-10-12 15:26:28 +0000320 x, y = os.pipe()
321 if mswindows:
322 pipe_buf = 512
323 else:
324 pipe_buf = os.fpathconf(x, "PC_PIPE_BUF")
325 os.close(x)
326 os.close(y)
327 p = subprocess.Popen([sys.executable, "-c",
Tim Peterse718f612004-10-12 21:51:32 +0000328 'import sys,os;'
Fredrik Lundh5b3687d2004-10-12 15:26:28 +0000329 'sys.stdout.write(sys.stdin.read(47));' \
330 'sys.stderr.write("xyz"*%d);' \
331 'sys.stdout.write(sys.stdin.read())' % pipe_buf],
Tim Peters3b01a702004-10-12 22:19:32 +0000332 stdin=subprocess.PIPE,
333 stdout=subprocess.PIPE,
334 stderr=subprocess.PIPE)
Guido van Rossumc9e363c2007-05-15 23:18:55 +0000335 string_to_write = b"abc"*pipe_buf
Fredrik Lundh5b3687d2004-10-12 15:26:28 +0000336 (stdout, stderr) = p.communicate(string_to_write)
337 self.assertEqual(stdout, string_to_write)
338
339 def test_writes_before_communicate(self):
Tim Peters7b759da2004-10-12 22:29:54 +0000340 # stdin.write before communicate()
Fredrik Lundh5b3687d2004-10-12 15:26:28 +0000341 p = subprocess.Popen([sys.executable, "-c",
342 'import sys,os;' \
343 'sys.stdout.write(sys.stdin.read())'],
Tim Peters3b01a702004-10-12 22:19:32 +0000344 stdin=subprocess.PIPE,
345 stdout=subprocess.PIPE,
346 stderr=subprocess.PIPE)
Guido van Rossumbb839ef2007-08-27 23:58:21 +0000347 p.stdin.write(b"banana")
348 (stdout, stderr) = p.communicate(b"split")
Guido van Rossumc9e363c2007-05-15 23:18:55 +0000349 self.assertEqual(stdout, b"bananasplit")
Tim Peters3761e8d2004-10-13 04:07:12 +0000350 self.assertEqual(remove_stderr_debug_decorations(stderr), "")
Tim Peterse718f612004-10-12 21:51:32 +0000351
Fredrik Lundh5b3687d2004-10-12 15:26:28 +0000352 def test_universal_newlines(self):
Fredrik Lundh5b3687d2004-10-12 15:26:28 +0000353 p = subprocess.Popen([sys.executable, "-c",
Tim Peters3b01a702004-10-12 22:19:32 +0000354 'import sys,os;' + SETBINARY +
355 'sys.stdout.write("line1\\n");'
356 'sys.stdout.flush();'
Guido van Rossumfa0054a2007-05-24 04:05:35 +0000357 'sys.stdout.write("line2\\n");'
Tim Peters3b01a702004-10-12 22:19:32 +0000358 'sys.stdout.flush();'
359 'sys.stdout.write("line3\\r\\n");'
360 'sys.stdout.flush();'
361 'sys.stdout.write("line4\\r");'
362 'sys.stdout.flush();'
Fredrik Lundh5b3687d2004-10-12 15:26:28 +0000363 'sys.stdout.write("\\nline5");'
Tim Peters3b01a702004-10-12 22:19:32 +0000364 'sys.stdout.flush();'
Fredrik Lundh5b3687d2004-10-12 15:26:28 +0000365 'sys.stdout.write("\\nline6");'],
366 stdout=subprocess.PIPE,
367 universal_newlines=1)
368 stdout = p.stdout.read()
Guido van Rossumc9e363c2007-05-15 23:18:55 +0000369 self.assertEqual(stdout, "line1\nline2\nline3\nline4\nline5\nline6")
Fredrik Lundh5b3687d2004-10-12 15:26:28 +0000370
371 def test_universal_newlines_communicate(self):
Tim Peters7b759da2004-10-12 22:29:54 +0000372 # universal newlines through communicate()
Fredrik Lundh5b3687d2004-10-12 15:26:28 +0000373 p = subprocess.Popen([sys.executable, "-c",
Tim Peters3b01a702004-10-12 22:19:32 +0000374 'import sys,os;' + SETBINARY +
375 'sys.stdout.write("line1\\n");'
376 'sys.stdout.flush();'
Guido van Rossumfa0054a2007-05-24 04:05:35 +0000377 'sys.stdout.write("line2\\n");'
Tim Peters3b01a702004-10-12 22:19:32 +0000378 'sys.stdout.flush();'
379 'sys.stdout.write("line3\\r\\n");'
380 'sys.stdout.flush();'
381 'sys.stdout.write("line4\\r");'
382 'sys.stdout.flush();'
Fredrik Lundh5b3687d2004-10-12 15:26:28 +0000383 'sys.stdout.write("\\nline5");'
Tim Peters3b01a702004-10-12 22:19:32 +0000384 'sys.stdout.flush();'
Fredrik Lundh5b3687d2004-10-12 15:26:28 +0000385 'sys.stdout.write("\\nline6");'],
386 stdout=subprocess.PIPE, stderr=subprocess.PIPE,
387 universal_newlines=1)
388 (stdout, stderr) = p.communicate()
Guido van Rossumfa0054a2007-05-24 04:05:35 +0000389 self.assertEqual(stdout, "line1\nline2\nline3\nline4\nline5\nline6")
Fredrik Lundh5b3687d2004-10-12 15:26:28 +0000390
391 def test_no_leaking(self):
Tim Peters7b759da2004-10-12 22:29:54 +0000392 # Make sure we leak no resources
Thomas Wouters0e3f5912006-08-11 14:57:12 +0000393 if not hasattr(test_support, "is_resource_enabled") \
394 or test_support.is_resource_enabled("subprocess") and not mswindows:
Peter Astrandf7f1bb72005-03-03 20:47:37 +0000395 max_handles = 1026 # too much for most UNIX systems
396 else:
Tim Peterseba28be2005-03-28 01:08:02 +0000397 max_handles = 65
Fredrik Lundh9e29fc52004-10-13 07:54:54 +0000398 for i in range(max_handles):
Tim Peters3b01a702004-10-12 22:19:32 +0000399 p = subprocess.Popen([sys.executable, "-c",
400 "import sys;sys.stdout.write(sys.stdin.read())"],
401 stdin=subprocess.PIPE,
402 stdout=subprocess.PIPE,
403 stderr=subprocess.PIPE)
Fredrik Lundh5b3687d2004-10-12 15:26:28 +0000404 data = p.communicate("lime")[0]
Guido van Rossumc9e363c2007-05-15 23:18:55 +0000405 self.assertEqual(data, b"lime")
Fredrik Lundh5b3687d2004-10-12 15:26:28 +0000406
407
408 def test_list2cmdline(self):
Fredrik Lundh5b3687d2004-10-12 15:26:28 +0000409 self.assertEqual(subprocess.list2cmdline(['a b c', 'd', 'e']),
410 '"a b c" d e')
411 self.assertEqual(subprocess.list2cmdline(['ab"c', '\\', 'd']),
412 'ab\\"c \\ d')
413 self.assertEqual(subprocess.list2cmdline(['a\\\\\\b', 'de fg', 'h']),
414 'a\\\\\\b "de fg" h')
415 self.assertEqual(subprocess.list2cmdline(['a\\"b', 'c', 'd']),
416 'a\\\\\\"b c d')
417 self.assertEqual(subprocess.list2cmdline(['a\\\\b c', 'd', 'e']),
418 '"a\\\\b c" d e')
419 self.assertEqual(subprocess.list2cmdline(['a\\\\b\\ c', 'd', 'e']),
420 '"a\\\\b\\ c" d e')
Thomas Woutersfc7bb8c2007-01-15 15:49:28 +0000421 self.assertEqual(subprocess.list2cmdline(['ab', '']),
422 'ab ""')
Fredrik Lundh5b3687d2004-10-12 15:26:28 +0000423
424
425 def test_poll(self):
Fredrik Lundh5b3687d2004-10-12 15:26:28 +0000426 p = subprocess.Popen([sys.executable,
Tim Peters29b6b4f2004-10-13 03:43:40 +0000427 "-c", "import time; time.sleep(1)"])
428 count = 0
429 while p.poll() is None:
430 time.sleep(0.1)
431 count += 1
432 # We expect that the poll loop probably went around about 10 times,
433 # but, based on system scheduling we can't control, it's possible
434 # poll() never returned None. It "should be" very rare that it
435 # didn't go around at least twice.
436 self.assert_(count >= 2)
Fredrik Lundh5b3687d2004-10-12 15:26:28 +0000437 # Subsequent invocations should just return the returncode
438 self.assertEqual(p.poll(), 0)
439
440
441 def test_wait(self):
Fredrik Lundh5b3687d2004-10-12 15:26:28 +0000442 p = subprocess.Popen([sys.executable,
443 "-c", "import time; time.sleep(2)"])
444 self.assertEqual(p.wait(), 0)
445 # Subsequent invocations should just return the returncode
446 self.assertEqual(p.wait(), 0)
Tim Peterse718f612004-10-12 21:51:32 +0000447
Peter Astrand738131d2004-11-30 21:04:45 +0000448
449 def test_invalid_bufsize(self):
450 # an invalid type of the bufsize argument should raise
451 # TypeError.
452 try:
453 subprocess.Popen([sys.executable, "-c", "pass"], "orange")
454 except TypeError:
455 pass
456 else:
457 self.fail("Expected TypeError")
458
Guido van Rossum46a05a72007-06-07 21:56:45 +0000459 def test_bufsize_is_none(self):
460 # bufsize=None should be the same as bufsize=0.
461 p = subprocess.Popen([sys.executable, "-c", "pass"], None)
462 self.assertEqual(p.wait(), 0)
463 # Again with keyword arg
464 p = subprocess.Popen([sys.executable, "-c", "pass"], bufsize=None)
465 self.assertEqual(p.wait(), 0)
466
Fredrik Lundh5b3687d2004-10-12 15:26:28 +0000467 #
468 # POSIX tests
469 #
470 if not mswindows:
471 def test_exceptions(self):
Guido van Rossumfa0054a2007-05-24 04:05:35 +0000472 # caught & re-raised exceptions
Fredrik Lundh5b3687d2004-10-12 15:26:28 +0000473 try:
474 p = subprocess.Popen([sys.executable, "-c", ""],
Guido van Rossumfa0054a2007-05-24 04:05:35 +0000475 cwd="/this/path/does/not/exist")
Guido van Rossumb940e112007-01-10 16:19:56 +0000476 except OSError as e:
Fredrik Lundh5b3687d2004-10-12 15:26:28 +0000477 # The attribute child_traceback should contain "os.chdir"
478 # somewhere.
479 self.assertNotEqual(e.child_traceback.find("os.chdir"), -1)
480 else:
481 self.fail("Expected OSError")
Tim Peterse718f612004-10-12 21:51:32 +0000482
Thomas Wouters0e3f5912006-08-11 14:57:12 +0000483 def _suppress_core_files(self):
484 """Try to prevent core files from being created.
485 Returns previous ulimit if successful, else None.
486 """
487 try:
488 import resource
489 old_limit = resource.getrlimit(resource.RLIMIT_CORE)
490 resource.setrlimit(resource.RLIMIT_CORE, (0,0))
491 return old_limit
492 except (ImportError, ValueError, resource.error):
493 return None
494
495 def _unsuppress_core_files(self, old_limit):
496 """Return core file behavior to default."""
497 if old_limit is None:
498 return
499 try:
500 import resource
501 resource.setrlimit(resource.RLIMIT_CORE, old_limit)
502 except (ImportError, ValueError, resource.error):
503 return
504
Fredrik Lundh5b3687d2004-10-12 15:26:28 +0000505 def test_run_abort(self):
Tim Peters7b759da2004-10-12 22:29:54 +0000506 # returncode handles signal termination
Thomas Wouters0e3f5912006-08-11 14:57:12 +0000507 old_limit = self._suppress_core_files()
508 try:
509 p = subprocess.Popen([sys.executable,
510 "-c", "import os; os.abort()"])
511 finally:
512 self._unsuppress_core_files(old_limit)
Fredrik Lundh5b3687d2004-10-12 15:26:28 +0000513 p.wait()
514 self.assertEqual(-p.returncode, signal.SIGABRT)
515
516 def test_preexec(self):
Tim Peters7b759da2004-10-12 22:29:54 +0000517 # preexec function
Fredrik Lundh5b3687d2004-10-12 15:26:28 +0000518 p = subprocess.Popen([sys.executable, "-c",
519 'import sys,os;' \
520 'sys.stdout.write(os.getenv("FRUIT"))'],
521 stdout=subprocess.PIPE,
522 preexec_fn=lambda: os.putenv("FRUIT", "apple"))
Guido van Rossumfa0054a2007-05-24 04:05:35 +0000523 self.assertEqual(p.stdout.read(), b"apple")
Fredrik Lundh5b3687d2004-10-12 15:26:28 +0000524
Fredrik Lundh5b3687d2004-10-12 15:26:28 +0000525 def test_args_string(self):
Tim Peters7b759da2004-10-12 22:29:54 +0000526 # args is a string
Fredrik Lundh5b3687d2004-10-12 15:26:28 +0000527 f, fname = self.mkstemp()
528 os.write(f, "#!/bin/sh\n")
Tim Peters3b01a702004-10-12 22:19:32 +0000529 os.write(f, "exec %s -c 'import sys; sys.exit(47)'\n" %
530 sys.executable)
Fredrik Lundh5b3687d2004-10-12 15:26:28 +0000531 os.close(f)
Guido van Rossumcd16bf62007-06-13 18:07:49 +0000532 os.chmod(fname, 0o700)
Fredrik Lundh5b3687d2004-10-12 15:26:28 +0000533 p = subprocess.Popen(fname)
534 p.wait()
Fredrik Lundh5b3687d2004-10-12 15:26:28 +0000535 os.remove(fname)
Peter Astrand2224be62004-11-17 20:06:35 +0000536 self.assertEqual(p.returncode, 47)
Fredrik Lundh5b3687d2004-10-12 15:26:28 +0000537
538 def test_invalid_args(self):
Tim Peters7b759da2004-10-12 22:29:54 +0000539 # invalid arguments should raise ValueError
Fredrik Lundh5b3687d2004-10-12 15:26:28 +0000540 self.assertRaises(ValueError, subprocess.call,
Tim Peters3b01a702004-10-12 22:19:32 +0000541 [sys.executable,
542 "-c", "import sys; sys.exit(47)"],
Fredrik Lundh5b3687d2004-10-12 15:26:28 +0000543 startupinfo=47)
544 self.assertRaises(ValueError, subprocess.call,
Tim Peters3b01a702004-10-12 22:19:32 +0000545 [sys.executable,
546 "-c", "import sys; sys.exit(47)"],
Fredrik Lundh5b3687d2004-10-12 15:26:28 +0000547 creationflags=47)
548
549 def test_shell_sequence(self):
Tim Peters7b759da2004-10-12 22:29:54 +0000550 # Run command through the shell (sequence)
Fredrik Lundh5b3687d2004-10-12 15:26:28 +0000551 newenv = os.environ.copy()
552 newenv["FRUIT"] = "apple"
553 p = subprocess.Popen(["echo $FRUIT"], shell=1,
554 stdout=subprocess.PIPE,
555 env=newenv)
Guido van Rossumfa0054a2007-05-24 04:05:35 +0000556 self.assertEqual(p.stdout.read().strip(b" \t\r\n\f"), b"apple")
Fredrik Lundh5b3687d2004-10-12 15:26:28 +0000557
558 def test_shell_string(self):
Tim Peters7b759da2004-10-12 22:29:54 +0000559 # Run command through the shell (string)
Fredrik Lundh5b3687d2004-10-12 15:26:28 +0000560 newenv = os.environ.copy()
561 newenv["FRUIT"] = "apple"
562 p = subprocess.Popen("echo $FRUIT", shell=1,
563 stdout=subprocess.PIPE,
564 env=newenv)
Guido van Rossumfa0054a2007-05-24 04:05:35 +0000565 self.assertEqual(p.stdout.read().strip(b" \t\r\n\f"), b"apple")
Fredrik Lundh5b3687d2004-10-12 15:26:28 +0000566
567 def test_call_string(self):
Tim Peters7b759da2004-10-12 22:29:54 +0000568 # call() function with string argument on UNIX
Fredrik Lundh5b3687d2004-10-12 15:26:28 +0000569 f, fname = self.mkstemp()
570 os.write(f, "#!/bin/sh\n")
Tim Peters3b01a702004-10-12 22:19:32 +0000571 os.write(f, "exec %s -c 'import sys; sys.exit(47)'\n" %
572 sys.executable)
Fredrik Lundh5b3687d2004-10-12 15:26:28 +0000573 os.close(f)
Guido van Rossumcd16bf62007-06-13 18:07:49 +0000574 os.chmod(fname, 0o700)
Fredrik Lundh5b3687d2004-10-12 15:26:28 +0000575 rc = subprocess.call(fname)
Peter Astrand2224be62004-11-17 20:06:35 +0000576 os.remove(fname)
Fredrik Lundh5b3687d2004-10-12 15:26:28 +0000577 self.assertEqual(rc, 47)
578
Tim Peterse718f612004-10-12 21:51:32 +0000579
Fredrik Lundh5b3687d2004-10-12 15:26:28 +0000580 #
581 # Windows tests
582 #
583 if mswindows:
584 def test_startupinfo(self):
Tim Peters7b759da2004-10-12 22:29:54 +0000585 # startupinfo argument
Fredrik Lundh5b3687d2004-10-12 15:26:28 +0000586 # We uses hardcoded constants, because we do not want to
Tim Peterse718f612004-10-12 21:51:32 +0000587 # depend on win32all.
Fredrik Lundh5b3687d2004-10-12 15:26:28 +0000588 STARTF_USESHOWWINDOW = 1
589 SW_MAXIMIZE = 3
590 startupinfo = subprocess.STARTUPINFO()
591 startupinfo.dwFlags = STARTF_USESHOWWINDOW
592 startupinfo.wShowWindow = SW_MAXIMIZE
593 # Since Python is a console process, it won't be affected
594 # by wShowWindow, but the argument should be silently
595 # ignored
596 subprocess.call([sys.executable, "-c", "import sys; sys.exit(0)"],
597 startupinfo=startupinfo)
598
599 def test_creationflags(self):
Tim Peters7b759da2004-10-12 22:29:54 +0000600 # creationflags argument
Fredrik Lundh5b3687d2004-10-12 15:26:28 +0000601 CREATE_NEW_CONSOLE = 16
Tim Peters876c4322004-10-13 03:21:35 +0000602 sys.stderr.write(" a DOS box should flash briefly ...\n")
Tim Peters3b01a702004-10-12 22:19:32 +0000603 subprocess.call(sys.executable +
Tim Peters876c4322004-10-13 03:21:35 +0000604 ' -c "import time; time.sleep(0.25)"',
Fredrik Lundh5b3687d2004-10-12 15:26:28 +0000605 creationflags=CREATE_NEW_CONSOLE)
606
607 def test_invalid_args(self):
Tim Peters7b759da2004-10-12 22:29:54 +0000608 # invalid arguments should raise ValueError
Fredrik Lundh5b3687d2004-10-12 15:26:28 +0000609 self.assertRaises(ValueError, subprocess.call,
Tim Peters3b01a702004-10-12 22:19:32 +0000610 [sys.executable,
611 "-c", "import sys; sys.exit(47)"],
Fredrik Lundh5b3687d2004-10-12 15:26:28 +0000612 preexec_fn=lambda: 1)
613 self.assertRaises(ValueError, subprocess.call,
Tim Peters3b01a702004-10-12 22:19:32 +0000614 [sys.executable,
615 "-c", "import sys; sys.exit(47)"],
Guido van Rossume7ba4952007-06-06 23:52:48 +0000616 stdout=subprocess.PIPE,
Fredrik Lundh5b3687d2004-10-12 15:26:28 +0000617 close_fds=True)
618
Guido van Rossume7ba4952007-06-06 23:52:48 +0000619 def test_close_fds(self):
620 # close file descriptors
621 rc = subprocess.call([sys.executable, "-c",
622 "import sys; sys.exit(47)"],
623 close_fds=True)
624 self.assertEqual(rc, 47)
625
Fredrik Lundh5b3687d2004-10-12 15:26:28 +0000626 def test_shell_sequence(self):
Tim Peters7b759da2004-10-12 22:29:54 +0000627 # Run command through the shell (sequence)
Fredrik Lundh5b3687d2004-10-12 15:26:28 +0000628 newenv = os.environ.copy()
629 newenv["FRUIT"] = "physalis"
630 p = subprocess.Popen(["set"], shell=1,
Tim Peterse718f612004-10-12 21:51:32 +0000631 stdout=subprocess.PIPE,
Fredrik Lundh5b3687d2004-10-12 15:26:28 +0000632 env=newenv)
633 self.assertNotEqual(p.stdout.read().find("physalis"), -1)
634
635 def test_shell_string(self):
Tim Peters7b759da2004-10-12 22:29:54 +0000636 # Run command through the shell (string)
Fredrik Lundh5b3687d2004-10-12 15:26:28 +0000637 newenv = os.environ.copy()
638 newenv["FRUIT"] = "physalis"
639 p = subprocess.Popen("set", shell=1,
Tim Peterse718f612004-10-12 21:51:32 +0000640 stdout=subprocess.PIPE,
Fredrik Lundh5b3687d2004-10-12 15:26:28 +0000641 env=newenv)
642 self.assertNotEqual(p.stdout.read().find("physalis"), -1)
643
644 def test_call_string(self):
Tim Peters7b759da2004-10-12 22:29:54 +0000645 # call() function with string argument on Windows
Tim Peters3b01a702004-10-12 22:19:32 +0000646 rc = subprocess.call(sys.executable +
647 ' -c "import sys; sys.exit(47)"')
Fredrik Lundh5b3687d2004-10-12 15:26:28 +0000648 self.assertEqual(rc, 47)
649
650
Fredrik Lundh5b3687d2004-10-12 15:26:28 +0000651def test_main():
652 test_support.run_unittest(ProcessTestCase)
Thomas Wouters0e3f5912006-08-11 14:57:12 +0000653 if hasattr(test_support, "reap_children"):
654 test_support.reap_children()
Fredrik Lundh5b3687d2004-10-12 15:26:28 +0000655
656if __name__ == "__main__":
657 test_main()