| 
									
										
										
										
											2014-02-01 22:49:59 +01:00
										 |  |  | from asyncio import subprocess | 
					
						
							| 
									
										
										
										
											2014-06-18 01:36:32 +02:00
										 |  |  | from asyncio import test_utils | 
					
						
							| 
									
										
										
										
											2014-02-01 22:49:59 +01:00
										 |  |  | import asyncio | 
					
						
							|  |  |  | import signal | 
					
						
							|  |  |  | import sys | 
					
						
							|  |  |  | import unittest | 
					
						
							|  |  |  | from test import support | 
					
						
							|  |  |  | if sys.platform != 'win32': | 
					
						
							|  |  |  |     from asyncio import unix_events | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | # Program blocking | 
					
						
							|  |  |  | PROGRAM_BLOCKED = [sys.executable, '-c', 'import time; time.sleep(3600)'] | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | # Program copying input to output | 
					
						
							|  |  |  | PROGRAM_CAT = [ | 
					
						
							|  |  |  |     sys.executable, '-c', | 
					
						
							|  |  |  |     ';'.join(('import sys', | 
					
						
							|  |  |  |               'data = sys.stdin.buffer.read()', | 
					
						
							|  |  |  |               'sys.stdout.buffer.write(data)'))] | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | class SubprocessMixin: | 
					
						
							| 
									
										
										
										
											2014-02-18 22:56:15 -05:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-02-01 22:49:59 +01:00
										 |  |  |     def test_stdin_stdout(self): | 
					
						
							|  |  |  |         args = PROGRAM_CAT | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         @asyncio.coroutine | 
					
						
							|  |  |  |         def run(data): | 
					
						
							|  |  |  |             proc = yield from asyncio.create_subprocess_exec( | 
					
						
							|  |  |  |                                           *args, | 
					
						
							|  |  |  |                                           stdin=subprocess.PIPE, | 
					
						
							|  |  |  |                                           stdout=subprocess.PIPE, | 
					
						
							|  |  |  |                                           loop=self.loop) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             # feed data | 
					
						
							|  |  |  |             proc.stdin.write(data) | 
					
						
							|  |  |  |             yield from proc.stdin.drain() | 
					
						
							|  |  |  |             proc.stdin.close() | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             # get output and exitcode | 
					
						
							|  |  |  |             data = yield from proc.stdout.read() | 
					
						
							|  |  |  |             exitcode = yield from proc.wait() | 
					
						
							|  |  |  |             return (exitcode, data) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         task = run(b'some data') | 
					
						
							| 
									
										
										
										
											2014-07-25 14:05:07 +02:00
										 |  |  |         task = asyncio.wait_for(task, 60.0, loop=self.loop) | 
					
						
							| 
									
										
										
										
											2014-02-01 22:49:59 +01:00
										 |  |  |         exitcode, stdout = self.loop.run_until_complete(task) | 
					
						
							|  |  |  |         self.assertEqual(exitcode, 0) | 
					
						
							|  |  |  |         self.assertEqual(stdout, b'some data') | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def test_communicate(self): | 
					
						
							|  |  |  |         args = PROGRAM_CAT | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         @asyncio.coroutine | 
					
						
							|  |  |  |         def run(data): | 
					
						
							|  |  |  |             proc = yield from asyncio.create_subprocess_exec( | 
					
						
							|  |  |  |                                           *args, | 
					
						
							|  |  |  |                                           stdin=subprocess.PIPE, | 
					
						
							|  |  |  |                                           stdout=subprocess.PIPE, | 
					
						
							|  |  |  |                                           loop=self.loop) | 
					
						
							|  |  |  |             stdout, stderr = yield from proc.communicate(data) | 
					
						
							|  |  |  |             return proc.returncode, stdout | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         task = run(b'some data') | 
					
						
							| 
									
										
										
										
											2014-07-25 14:05:07 +02:00
										 |  |  |         task = asyncio.wait_for(task, 60.0, loop=self.loop) | 
					
						
							| 
									
										
										
										
											2014-02-01 22:49:59 +01:00
										 |  |  |         exitcode, stdout = self.loop.run_until_complete(task) | 
					
						
							|  |  |  |         self.assertEqual(exitcode, 0) | 
					
						
							|  |  |  |         self.assertEqual(stdout, b'some data') | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def test_shell(self): | 
					
						
							|  |  |  |         create = asyncio.create_subprocess_shell('exit 7', | 
					
						
							|  |  |  |                                                  loop=self.loop) | 
					
						
							|  |  |  |         proc = self.loop.run_until_complete(create) | 
					
						
							|  |  |  |         exitcode = self.loop.run_until_complete(proc.wait()) | 
					
						
							|  |  |  |         self.assertEqual(exitcode, 7) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def test_start_new_session(self): | 
					
						
							|  |  |  |         # start the new process in a new session | 
					
						
							|  |  |  |         create = asyncio.create_subprocess_shell('exit 8', | 
					
						
							|  |  |  |                                                  start_new_session=True, | 
					
						
							|  |  |  |                                                  loop=self.loop) | 
					
						
							|  |  |  |         proc = self.loop.run_until_complete(create) | 
					
						
							|  |  |  |         exitcode = self.loop.run_until_complete(proc.wait()) | 
					
						
							|  |  |  |         self.assertEqual(exitcode, 8) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def test_kill(self): | 
					
						
							|  |  |  |         args = PROGRAM_BLOCKED | 
					
						
							|  |  |  |         create = asyncio.create_subprocess_exec(*args, loop=self.loop) | 
					
						
							|  |  |  |         proc = self.loop.run_until_complete(create) | 
					
						
							|  |  |  |         proc.kill() | 
					
						
							|  |  |  |         returncode = self.loop.run_until_complete(proc.wait()) | 
					
						
							|  |  |  |         if sys.platform == 'win32': | 
					
						
							|  |  |  |             self.assertIsInstance(returncode, int) | 
					
						
							|  |  |  |             # expect 1 but sometimes get 0 | 
					
						
							|  |  |  |         else: | 
					
						
							|  |  |  |             self.assertEqual(-signal.SIGKILL, returncode) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def test_terminate(self): | 
					
						
							|  |  |  |         args = PROGRAM_BLOCKED | 
					
						
							|  |  |  |         create = asyncio.create_subprocess_exec(*args, loop=self.loop) | 
					
						
							|  |  |  |         proc = self.loop.run_until_complete(create) | 
					
						
							|  |  |  |         proc.terminate() | 
					
						
							|  |  |  |         returncode = self.loop.run_until_complete(proc.wait()) | 
					
						
							|  |  |  |         if sys.platform == 'win32': | 
					
						
							|  |  |  |             self.assertIsInstance(returncode, int) | 
					
						
							|  |  |  |             # expect 1 but sometimes get 0 | 
					
						
							|  |  |  |         else: | 
					
						
							|  |  |  |             self.assertEqual(-signal.SIGTERM, returncode) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     @unittest.skipIf(sys.platform == 'win32', "Don't have SIGHUP") | 
					
						
							|  |  |  |     def test_send_signal(self): | 
					
						
							| 
									
										
										
										
											2014-07-17 23:49:11 +02:00
										 |  |  |         code = 'import time; print("sleeping", flush=True); time.sleep(3600)' | 
					
						
							|  |  |  |         args = [sys.executable, '-c', code] | 
					
						
							|  |  |  |         create = asyncio.create_subprocess_exec(*args, loop=self.loop, stdout=subprocess.PIPE) | 
					
						
							| 
									
										
										
										
											2014-02-01 22:49:59 +01:00
										 |  |  |         proc = self.loop.run_until_complete(create) | 
					
						
							| 
									
										
										
										
											2014-07-17 23:49:11 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  |         @asyncio.coroutine | 
					
						
							|  |  |  |         def send_signal(proc): | 
					
						
							|  |  |  |             # basic synchronization to wait until the program is sleeping | 
					
						
							|  |  |  |             line = yield from proc.stdout.readline() | 
					
						
							|  |  |  |             self.assertEqual(line, b'sleeping\n') | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             proc.send_signal(signal.SIGHUP) | 
					
						
							|  |  |  |             returncode = (yield from proc.wait()) | 
					
						
							|  |  |  |             return returncode | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         returncode = self.loop.run_until_complete(send_signal(proc)) | 
					
						
							| 
									
										
										
										
											2014-02-01 22:49:59 +01:00
										 |  |  |         self.assertEqual(-signal.SIGHUP, returncode) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-07-17 12:25:27 +02:00
										 |  |  |     def prepare_broken_pipe_test(self): | 
					
						
							|  |  |  |         # buffer large enough to feed the whole pipe buffer | 
					
						
							| 
									
										
										
										
											2014-02-01 22:49:59 +01:00
										 |  |  |         large_data = b'x' * support.PIPE_MAX_SIZE | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-07-17 12:25:27 +02:00
										 |  |  |         # the program ends before the stdin can be feeded | 
					
						
							| 
									
										
										
										
											2014-02-01 22:49:59 +01:00
										 |  |  |         create = asyncio.create_subprocess_exec( | 
					
						
							| 
									
										
										
										
											2014-07-17 12:25:27 +02:00
										 |  |  |                              sys.executable, '-c', 'pass', | 
					
						
							| 
									
										
										
										
											2014-02-01 22:49:59 +01:00
										 |  |  |                              stdin=subprocess.PIPE, | 
					
						
							|  |  |  |                              loop=self.loop) | 
					
						
							|  |  |  |         proc = self.loop.run_until_complete(create) | 
					
						
							| 
									
										
										
										
											2014-07-17 12:25:27 +02:00
										 |  |  |         return (proc, large_data) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def test_stdin_broken_pipe(self): | 
					
						
							|  |  |  |         proc, large_data = self.prepare_broken_pipe_test() | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-07-21 16:23:33 +02:00
										 |  |  |         @asyncio.coroutine | 
					
						
							|  |  |  |         def write_stdin(proc, data): | 
					
						
							|  |  |  |             proc.stdin.write(data) | 
					
						
							|  |  |  |             yield from proc.stdin.drain() | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         coro = write_stdin(proc, large_data) | 
					
						
							| 
									
										
										
										
											2014-07-17 14:01:14 +02:00
										 |  |  |         # drain() must raise BrokenPipeError or ConnectionResetError | 
					
						
							|  |  |  |         self.assertRaises((BrokenPipeError, ConnectionResetError), | 
					
						
							| 
									
										
										
										
											2014-07-21 16:23:33 +02:00
										 |  |  |                           self.loop.run_until_complete, coro) | 
					
						
							| 
									
										
										
										
											2014-07-17 12:25:27 +02:00
										 |  |  |         self.loop.run_until_complete(proc.wait()) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def test_communicate_ignore_broken_pipe(self): | 
					
						
							|  |  |  |         proc, large_data = self.prepare_broken_pipe_test() | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         # communicate() must ignore BrokenPipeError when feeding stdin | 
					
						
							|  |  |  |         self.loop.run_until_complete(proc.communicate(large_data)) | 
					
						
							| 
									
										
										
										
											2014-02-01 22:49:59 +01:00
										 |  |  |         self.loop.run_until_complete(proc.wait()) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | if sys.platform != 'win32': | 
					
						
							|  |  |  |     # Unix | 
					
						
							|  |  |  |     class SubprocessWatcherMixin(SubprocessMixin): | 
					
						
							| 
									
										
										
										
											2014-02-18 22:56:15 -05:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-02-01 22:49:59 +01:00
										 |  |  |         Watcher = None | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         def setUp(self): | 
					
						
							|  |  |  |             policy = asyncio.get_event_loop_policy() | 
					
						
							|  |  |  |             self.loop = policy.new_event_loop() | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-06-23 00:12:14 +02:00
										 |  |  |             # ensure that the event loop is passed explicitly in asyncio | 
					
						
							| 
									
										
										
										
											2014-02-01 22:49:59 +01:00
										 |  |  |             policy.set_event_loop(None) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             watcher = self.Watcher() | 
					
						
							|  |  |  |             watcher.attach_loop(self.loop) | 
					
						
							|  |  |  |             policy.set_child_watcher(watcher) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         def tearDown(self): | 
					
						
							|  |  |  |             policy = asyncio.get_event_loop_policy() | 
					
						
							|  |  |  |             policy.set_child_watcher(None) | 
					
						
							|  |  |  |             self.loop.close() | 
					
						
							| 
									
										
										
										
											2014-06-18 01:36:32 +02:00
										 |  |  |             super().tearDown() | 
					
						
							| 
									
										
										
										
											2014-02-01 22:49:59 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-02-18 22:56:15 -05:00
										 |  |  |     class SubprocessSafeWatcherTests(SubprocessWatcherMixin, | 
					
						
							| 
									
										
										
										
											2014-06-18 01:36:32 +02:00
										 |  |  |                                      test_utils.TestCase): | 
					
						
							| 
									
										
										
										
											2014-02-18 22:56:15 -05:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-02-01 22:49:59 +01:00
										 |  |  |         Watcher = unix_events.SafeChildWatcher | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-02-18 22:56:15 -05:00
										 |  |  |     class SubprocessFastWatcherTests(SubprocessWatcherMixin, | 
					
						
							| 
									
										
										
										
											2014-06-18 01:36:32 +02:00
										 |  |  |                                      test_utils.TestCase): | 
					
						
							| 
									
										
										
										
											2014-02-18 22:56:15 -05:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-02-01 22:49:59 +01:00
										 |  |  |         Watcher = unix_events.FastChildWatcher | 
					
						
							| 
									
										
										
										
											2014-02-18 22:56:15 -05:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-02-01 22:49:59 +01:00
										 |  |  | else: | 
					
						
							|  |  |  |     # Windows | 
					
						
							| 
									
										
										
										
											2014-06-18 01:36:32 +02:00
										 |  |  |     class SubprocessProactorTests(SubprocessMixin, test_utils.TestCase): | 
					
						
							| 
									
										
										
										
											2014-02-18 22:56:15 -05:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-02-01 22:49:59 +01:00
										 |  |  |         def setUp(self): | 
					
						
							|  |  |  |             policy = asyncio.get_event_loop_policy() | 
					
						
							|  |  |  |             self.loop = asyncio.ProactorEventLoop() | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-06-23 00:12:14 +02:00
										 |  |  |             # ensure that the event loop is passed explicitly in asyncio | 
					
						
							| 
									
										
										
										
											2014-02-01 22:49:59 +01:00
										 |  |  |             policy.set_event_loop(None) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         def tearDown(self): | 
					
						
							|  |  |  |             policy = asyncio.get_event_loop_policy() | 
					
						
							|  |  |  |             self.loop.close() | 
					
						
							|  |  |  |             policy.set_event_loop(None) | 
					
						
							| 
									
										
										
										
											2014-06-18 01:36:32 +02:00
										 |  |  |             super().tearDown() | 
					
						
							| 
									
										
										
										
											2014-02-01 22:49:59 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | if __name__ == '__main__': | 
					
						
							|  |  |  |     unittest.main() |