blob: 02b9e89f70915aba4e76e65563313c01ad22ef05 [file] [log] [blame]
Guido van Rossum0016e1d2013-10-30 14:56:49 -07001import collections
2import subprocess
Victor Stinner978a9af2015-01-29 17:50:58 +01003import sys
4import warnings
Guido van Rossum0016e1d2013-10-30 14:56:49 -07005
Victor Stinner47cd10d2015-01-30 00:05:19 +01006from . import futures
Guido van Rossum0016e1d2013-10-30 14:56:49 -07007from . import protocols
Guido van Rossum0016e1d2013-10-30 14:56:49 -07008from . import transports
Victor Stinnerf951d282014-06-29 00:46:45 +02009from .coroutines import coroutine
Victor Stinneracdb7822014-07-14 18:33:40 +020010from .log import logger
Guido van Rossum0016e1d2013-10-30 14:56:49 -070011
12
Guido van Rossum0016e1d2013-10-30 14:56:49 -070013class BaseSubprocessTransport(transports.SubprocessTransport):
14
15 def __init__(self, loop, protocol, args, shell,
16 stdin, stdout, stderr, bufsize,
Victor Stinner47cd10d2015-01-30 00:05:19 +010017 waiter=None, extra=None, **kwargs):
Guido van Rossum0016e1d2013-10-30 14:56:49 -070018 super().__init__(extra)
Victor Stinner978a9af2015-01-29 17:50:58 +010019 self._closed = False
Guido van Rossum0016e1d2013-10-30 14:56:49 -070020 self._protocol = protocol
21 self._loop = loop
Victor Stinner47cd10d2015-01-30 00:05:19 +010022 self._proc = None
Victor Stinneracdb7822014-07-14 18:33:40 +020023 self._pid = None
Victor Stinner47cd10d2015-01-30 00:05:19 +010024 self._returncode = None
25 self._exit_waiters = []
26 self._pending_calls = collections.deque()
Guido van Rossum0016e1d2013-10-30 14:56:49 -070027 self._pipes = {}
Victor Stinner47cd10d2015-01-30 00:05:19 +010028 self._finished = False
29
Guido van Rossum0016e1d2013-10-30 14:56:49 -070030 if stdin == subprocess.PIPE:
Victor Stinner915bcb02014-02-01 22:49:59 +010031 self._pipes[0] = None
Guido van Rossum0016e1d2013-10-30 14:56:49 -070032 if stdout == subprocess.PIPE:
Victor Stinner915bcb02014-02-01 22:49:59 +010033 self._pipes[1] = None
Guido van Rossum0016e1d2013-10-30 14:56:49 -070034 if stderr == subprocess.PIPE:
Victor Stinner915bcb02014-02-01 22:49:59 +010035 self._pipes[2] = None
Victor Stinner47cd10d2015-01-30 00:05:19 +010036
37 # Create the child process: set the _proc attribute
Guido van Rossum0016e1d2013-10-30 14:56:49 -070038 self._start(args=args, shell=shell, stdin=stdin, stdout=stdout,
39 stderr=stderr, bufsize=bufsize, **kwargs)
Victor Stinneracdb7822014-07-14 18:33:40 +020040 self._pid = self._proc.pid
Guido van Rossum0016e1d2013-10-30 14:56:49 -070041 self._extra['subprocess'] = self._proc
Victor Stinner47cd10d2015-01-30 00:05:19 +010042
Victor Stinneracdb7822014-07-14 18:33:40 +020043 if self._loop.get_debug():
44 if isinstance(args, (bytes, str)):
45 program = args
46 else:
47 program = args[0]
48 logger.debug('process %r created: pid %s',
49 program, self._pid)
50
Victor Stinner47cd10d2015-01-30 00:05:19 +010051 self._loop.create_task(self._connect_pipes(waiter))
52
Victor Stinneracdb7822014-07-14 18:33:40 +020053 def __repr__(self):
Victor Stinner978a9af2015-01-29 17:50:58 +010054 info = [self.__class__.__name__]
55 if self._closed:
56 info.append('closed')
57 info.append('pid=%s' % self._pid)
Victor Stinneracdb7822014-07-14 18:33:40 +020058 if self._returncode is not None:
59 info.append('returncode=%s' % self._returncode)
60
61 stdin = self._pipes.get(0)
62 if stdin is not None:
63 info.append('stdin=%s' % stdin.pipe)
64
65 stdout = self._pipes.get(1)
66 stderr = self._pipes.get(2)
67 if stdout is not None and stderr is stdout:
68 info.append('stdout=stderr=%s' % stdout.pipe)
69 else:
70 if stdout is not None:
71 info.append('stdout=%s' % stdout.pipe)
72 if stderr is not None:
73 info.append('stderr=%s' % stderr.pipe)
74
75 return '<%s>' % ' '.join(info)
Guido van Rossum0016e1d2013-10-30 14:56:49 -070076
77 def _start(self, args, shell, stdin, stdout, stderr, bufsize, **kwargs):
78 raise NotImplementedError
79
80 def _make_write_subprocess_pipe_proto(self, fd):
81 raise NotImplementedError
82
83 def _make_read_subprocess_pipe_proto(self, fd):
84 raise NotImplementedError
85
86 def close(self):
Victor Stinnerf2e43cb2015-01-30 01:20:44 +010087 if self._closed:
88 return
Victor Stinner978a9af2015-01-29 17:50:58 +010089 self._closed = True
Victor Stinner47cd10d2015-01-30 00:05:19 +010090
Guido van Rossum0016e1d2013-10-30 14:56:49 -070091 for proto in self._pipes.values():
Victor Stinner29ad0112015-01-15 00:04:21 +010092 if proto is None:
93 continue
Guido van Rossum0016e1d2013-10-30 14:56:49 -070094 proto.pipe.close()
Victor Stinner47cd10d2015-01-30 00:05:19 +010095
96 if self._proc is not None and self._returncode is None:
97 if self._loop.get_debug():
98 logger.warning('Close running child process: kill %r', self)
99
100 try:
101 self._proc.kill()
102 except ProcessLookupError:
103 pass
104
Victor Stinnerf2e43cb2015-01-30 01:20:44 +0100105 # Don't clear the _proc reference yet: _post_init() may still run
Guido van Rossum0016e1d2013-10-30 14:56:49 -0700106
Victor Stinner978a9af2015-01-29 17:50:58 +0100107 # On Python 3.3 and older, objects with a destructor part of a reference
108 # cycle are never destroyed. It's not more the case on Python 3.4 thanks
109 # to the PEP 442.
110 if sys.version_info >= (3, 4):
111 def __del__(self):
112 if not self._closed:
113 warnings.warn("unclosed transport %r" % self, ResourceWarning)
114 self.close()
115
Guido van Rossum0016e1d2013-10-30 14:56:49 -0700116 def get_pid(self):
Victor Stinneracdb7822014-07-14 18:33:40 +0200117 return self._pid
Guido van Rossum0016e1d2013-10-30 14:56:49 -0700118
119 def get_returncode(self):
120 return self._returncode
121
122 def get_pipe_transport(self, fd):
123 if fd in self._pipes:
124 return self._pipes[fd].pipe
125 else:
126 return None
127
Victor Stinner47cd10d2015-01-30 00:05:19 +0100128 def _check_proc(self):
Victor Stinner47cd10d2015-01-30 00:05:19 +0100129 if self._proc is None:
130 raise ProcessLookupError()
131
Guido van Rossum0016e1d2013-10-30 14:56:49 -0700132 def send_signal(self, signal):
Victor Stinner47cd10d2015-01-30 00:05:19 +0100133 self._check_proc()
Guido van Rossum0016e1d2013-10-30 14:56:49 -0700134 self._proc.send_signal(signal)
135
136 def terminate(self):
Victor Stinner47cd10d2015-01-30 00:05:19 +0100137 self._check_proc()
Guido van Rossum0016e1d2013-10-30 14:56:49 -0700138 self._proc.terminate()
139
140 def kill(self):
Victor Stinner47cd10d2015-01-30 00:05:19 +0100141 self._check_proc()
Guido van Rossum0016e1d2013-10-30 14:56:49 -0700142 self._proc.kill()
143
Victor Stinnerf951d282014-06-29 00:46:45 +0200144 @coroutine
Victor Stinner47cd10d2015-01-30 00:05:19 +0100145 def _connect_pipes(self, waiter):
Victor Stinnerf651a602015-01-14 02:10:33 +0100146 try:
147 proc = self._proc
148 loop = self._loop
Victor Stinner47cd10d2015-01-30 00:05:19 +0100149
Victor Stinnerf651a602015-01-14 02:10:33 +0100150 if proc.stdin is not None:
151 _, pipe = yield from loop.connect_write_pipe(
152 lambda: WriteSubprocessPipeProto(self, 0),
153 proc.stdin)
154 self._pipes[0] = pipe
Victor Stinner47cd10d2015-01-30 00:05:19 +0100155
Victor Stinnerf651a602015-01-14 02:10:33 +0100156 if proc.stdout is not None:
157 _, pipe = yield from loop.connect_read_pipe(
158 lambda: ReadSubprocessPipeProto(self, 1),
159 proc.stdout)
160 self._pipes[1] = pipe
Victor Stinner47cd10d2015-01-30 00:05:19 +0100161
Victor Stinnerf651a602015-01-14 02:10:33 +0100162 if proc.stderr is not None:
163 _, pipe = yield from loop.connect_read_pipe(
164 lambda: ReadSubprocessPipeProto(self, 2),
165 proc.stderr)
166 self._pipes[2] = pipe
Victor Stinneraaabc4f2014-01-29 14:22:56 -0800167
Victor Stinnerf651a602015-01-14 02:10:33 +0100168 assert self._pending_calls is not None
Victor Stinneraaabc4f2014-01-29 14:22:56 -0800169
Victor Stinner47cd10d2015-01-30 00:05:19 +0100170 loop.call_soon(self._protocol.connection_made, self)
Victor Stinnerf651a602015-01-14 02:10:33 +0100171 for callback, data in self._pending_calls:
Victor Stinner47cd10d2015-01-30 00:05:19 +0100172 loop.call_soon(callback, *data)
Victor Stinnerf651a602015-01-14 02:10:33 +0100173 self._pending_calls = None
Victor Stinner47cd10d2015-01-30 00:05:19 +0100174 except Exception as exc:
175 if waiter is not None and not waiter.cancelled():
176 waiter.set_exception(exc)
177 else:
178 if waiter is not None and not waiter.cancelled():
179 waiter.set_result(None)
Guido van Rossum0016e1d2013-10-30 14:56:49 -0700180
181 def _call(self, cb, *data):
182 if self._pending_calls is not None:
183 self._pending_calls.append((cb, data))
184 else:
185 self._loop.call_soon(cb, *data)
186
Guido van Rossum0016e1d2013-10-30 14:56:49 -0700187 def _pipe_connection_lost(self, fd, exc):
188 self._call(self._protocol.pipe_connection_lost, fd, exc)
189 self._try_finish()
190
191 def _pipe_data_received(self, fd, data):
192 self._call(self._protocol.pipe_data_received, fd, data)
193
194 def _process_exited(self, returncode):
195 assert returncode is not None, returncode
196 assert self._returncode is None, self._returncode
Victor Stinneracdb7822014-07-14 18:33:40 +0200197 if self._loop.get_debug():
198 logger.info('%r exited with return code %r',
199 self, returncode)
Guido van Rossum0016e1d2013-10-30 14:56:49 -0700200 self._returncode = returncode
Guido van Rossum0016e1d2013-10-30 14:56:49 -0700201 self._call(self._protocol.process_exited)
202 self._try_finish()
203
Victor Stinner47cd10d2015-01-30 00:05:19 +0100204 # wake up futures waiting for wait()
205 for waiter in self._exit_waiters:
206 if not waiter.cancelled():
207 waiter.set_result(returncode)
208 self._exit_waiters = None
209
Victor Stinner1241ecc2015-01-30 00:16:14 +0100210 def _wait(self):
Victor Stinner47cd10d2015-01-30 00:05:19 +0100211 """Wait until the process exit and return the process return code.
212
213 This method is a coroutine."""
214 if self._returncode is not None:
215 return self._returncode
216
217 waiter = futures.Future(loop=self._loop)
218 self._exit_waiters.append(waiter)
219 return (yield from waiter)
220
Guido van Rossum0016e1d2013-10-30 14:56:49 -0700221 def _try_finish(self):
222 assert not self._finished
223 if self._returncode is None:
224 return
225 if all(p is not None and p.disconnected
226 for p in self._pipes.values()):
227 self._finished = True
Victor Stinner1b9763d2014-12-18 23:47:27 +0100228 self._call(self._call_connection_lost, None)
Guido van Rossum0016e1d2013-10-30 14:56:49 -0700229
230 def _call_connection_lost(self, exc):
231 try:
232 self._protocol.connection_lost(exc)
233 finally:
Victor Stinner47cd10d2015-01-30 00:05:19 +0100234 self._loop = None
Guido van Rossum0016e1d2013-10-30 14:56:49 -0700235 self._proc = None
236 self._protocol = None
Guido van Rossum0016e1d2013-10-30 14:56:49 -0700237
238
239class WriteSubprocessPipeProto(protocols.BaseProtocol):
Guido van Rossum0016e1d2013-10-30 14:56:49 -0700240
241 def __init__(self, proc, fd):
242 self.proc = proc
243 self.fd = fd
Victor Stinneraaabc4f2014-01-29 14:22:56 -0800244 self.pipe = None
Guido van Rossum0016e1d2013-10-30 14:56:49 -0700245 self.disconnected = False
Guido van Rossum0016e1d2013-10-30 14:56:49 -0700246
247 def connection_made(self, transport):
Guido van Rossum0016e1d2013-10-30 14:56:49 -0700248 self.pipe = transport
Guido van Rossum0016e1d2013-10-30 14:56:49 -0700249
Victor Stinneracdb7822014-07-14 18:33:40 +0200250 def __repr__(self):
251 return ('<%s fd=%s pipe=%r>'
252 % (self.__class__.__name__, self.fd, self.pipe))
253
Guido van Rossum0016e1d2013-10-30 14:56:49 -0700254 def connection_lost(self, exc):
255 self.disconnected = True
256 self.proc._pipe_connection_lost(self.fd, exc)
Victor Stinner587feb12015-01-09 21:34:27 +0100257 self.proc = None
Guido van Rossum0016e1d2013-10-30 14:56:49 -0700258
Guido van Rossum1e9a4462014-01-29 14:28:15 -0800259 def pause_writing(self):
260 self.proc._protocol.pause_writing()
261
262 def resume_writing(self):
263 self.proc._protocol.resume_writing()
Guido van Rossum0016e1d2013-10-30 14:56:49 -0700264
265
266class ReadSubprocessPipeProto(WriteSubprocessPipeProto,
267 protocols.Protocol):
268
269 def data_received(self, data):
270 self.proc._pipe_data_received(self.fd, data)