blob: e6bec71d6c7dac3905e1841e218b7f13dee23cd8 [file] [log] [blame]
Yury Selivanov6370f342017-12-10 18:36:12 -05001__all__ = 'create_subprocess_exec', 'create_subprocess_shell'
Victor Stinner915bcb02014-02-01 22:49:59 +01002
Victor Stinner915bcb02014-02-01 22:49:59 +01003import subprocess
Andrew Svetlovad4ed872019-05-06 22:52:11 -04004import warnings
Victor Stinner915bcb02014-02-01 22:49:59 +01005
6from . import events
Victor Stinner915bcb02014-02-01 22:49:59 +01007from . import protocols
8from . import streams
9from . import tasks
Victor Stinneracdb7822014-07-14 18:33:40 +020010from .log import logger
Victor Stinner915bcb02014-02-01 22:49:59 +010011
12
13PIPE = subprocess.PIPE
14STDOUT = subprocess.STDOUT
15DEVNULL = subprocess.DEVNULL
16
17
18class SubprocessStreamProtocol(streams.FlowControlMixin,
19 protocols.SubprocessProtocol):
20 """Like StreamReaderProtocol, but for a subprocess."""
21
Andrew Svetlovad4ed872019-05-06 22:52:11 -040022 def __init__(self, limit, loop, *, _asyncio_internal=False):
23 super().__init__(loop=loop, _asyncio_internal=_asyncio_internal)
Victor Stinner915bcb02014-02-01 22:49:59 +010024 self._limit = limit
25 self.stdin = self.stdout = self.stderr = None
Victor Stinner915bcb02014-02-01 22:49:59 +010026 self._transport = None
Seth M. Larson481cb702017-03-02 22:21:18 -060027 self._process_exited = False
28 self._pipe_fds = []
Andrew Svetlov1cc0ee72019-05-07 16:53:19 -040029 self._stdin_closed = self._loop.create_future()
Andrew Svetlov23b4b692019-05-27 22:56:22 +030030 self._stdout_closed = self._loop.create_future()
31 self._stderr_closed = self._loop.create_future()
Victor Stinner915bcb02014-02-01 22:49:59 +010032
Victor Stinneracdb7822014-07-14 18:33:40 +020033 def __repr__(self):
34 info = [self.__class__.__name__]
35 if self.stdin is not None:
Yury Selivanov6370f342017-12-10 18:36:12 -050036 info.append(f'stdin={self.stdin!r}')
Victor Stinneracdb7822014-07-14 18:33:40 +020037 if self.stdout is not None:
Yury Selivanov6370f342017-12-10 18:36:12 -050038 info.append(f'stdout={self.stdout!r}')
Victor Stinneracdb7822014-07-14 18:33:40 +020039 if self.stderr is not None:
Yury Selivanov6370f342017-12-10 18:36:12 -050040 info.append(f'stderr={self.stderr!r}')
41 return '<{}>'.format(' '.join(info))
Victor Stinneracdb7822014-07-14 18:33:40 +020042
Victor Stinner915bcb02014-02-01 22:49:59 +010043 def connection_made(self, transport):
44 self._transport = transport
Victor Stinner5ef586f2014-11-25 17:20:33 +010045 stdout_transport = transport.get_pipe_transport(1)
46 if stdout_transport is not None:
Andrew Svetlov23b4b692019-05-27 22:56:22 +030047 self.stdout = streams.Stream(mode=streams.StreamMode.READ,
48 transport=stdout_transport,
49 protocol=self,
50 limit=self._limit,
51 loop=self._loop,
52 _asyncio_internal=True)
Victor Stinner5ef586f2014-11-25 17:20:33 +010053 self.stdout.set_transport(stdout_transport)
Seth M. Larson481cb702017-03-02 22:21:18 -060054 self._pipe_fds.append(1)
Victor Stinner5ef586f2014-11-25 17:20:33 +010055
56 stderr_transport = transport.get_pipe_transport(2)
57 if stderr_transport is not None:
Andrew Svetlov23b4b692019-05-27 22:56:22 +030058 self.stderr = streams.Stream(mode=streams.StreamMode.READ,
59 transport=stderr_transport,
60 protocol=self,
61 limit=self._limit,
62 loop=self._loop,
63 _asyncio_internal=True)
Victor Stinner5ef586f2014-11-25 17:20:33 +010064 self.stderr.set_transport(stderr_transport)
Seth M. Larson481cb702017-03-02 22:21:18 -060065 self._pipe_fds.append(2)
Victor Stinner5ef586f2014-11-25 17:20:33 +010066
67 stdin_transport = transport.get_pipe_transport(0)
68 if stdin_transport is not None:
Andrew Svetlov23b4b692019-05-27 22:56:22 +030069 self.stdin = streams.Stream(mode=streams.StreamMode.WRITE,
70 transport=stdin_transport,
71 protocol=self,
72 loop=self._loop,
73 _asyncio_internal=True)
Victor Stinnerf651a602015-01-14 02:10:33 +010074
Victor Stinner915bcb02014-02-01 22:49:59 +010075 def pipe_data_received(self, fd, data):
76 if fd == 1:
77 reader = self.stdout
78 elif fd == 2:
79 reader = self.stderr
80 else:
81 reader = None
82 if reader is not None:
83 reader.feed_data(data)
84
85 def pipe_connection_lost(self, fd, exc):
86 if fd == 0:
87 pipe = self.stdin
88 if pipe is not None:
89 pipe.close()
90 self.connection_lost(exc)
Andrew Svetlov1cc0ee72019-05-07 16:53:19 -040091 if exc is None:
92 self._stdin_closed.set_result(None)
93 else:
94 self._stdin_closed.set_exception(exc)
Victor Stinner915bcb02014-02-01 22:49:59 +010095 return
96 if fd == 1:
97 reader = self.stdout
98 elif fd == 2:
99 reader = self.stderr
100 else:
101 reader = None
Yury Selivanov6370f342017-12-10 18:36:12 -0500102 if reader is not None:
Victor Stinner915bcb02014-02-01 22:49:59 +0100103 if exc is None:
104 reader.feed_eof()
105 else:
106 reader.set_exception(exc)
Yury Selivanov2f156452017-03-02 23:25:31 -0500107
Seth M. Larson481cb702017-03-02 22:21:18 -0600108 if fd in self._pipe_fds:
109 self._pipe_fds.remove(fd)
110 self._maybe_close_transport()
Victor Stinner915bcb02014-02-01 22:49:59 +0100111
112 def process_exited(self):
Seth M. Larson481cb702017-03-02 22:21:18 -0600113 self._process_exited = True
114 self._maybe_close_transport()
Yury Selivanov2f156452017-03-02 23:25:31 -0500115
Seth M. Larson481cb702017-03-02 22:21:18 -0600116 def _maybe_close_transport(self):
117 if len(self._pipe_fds) == 0 and self._process_exited:
118 self._transport.close()
119 self._transport = None
Victor Stinner791009b2015-01-15 13:16:02 +0100120
Andrew Svetlov1cc0ee72019-05-07 16:53:19 -0400121 def _get_close_waiter(self, stream):
122 if stream is self.stdin:
123 return self._stdin_closed
Andrew Svetlov23b4b692019-05-27 22:56:22 +0300124 elif stream is self.stdout:
125 return self._stdout_closed
126 elif stream is self.stderr:
127 return self._stderr_closed
Andrew Svetlov1cc0ee72019-05-07 16:53:19 -0400128
Victor Stinner915bcb02014-02-01 22:49:59 +0100129
130class Process:
Andrew Svetlovad4ed872019-05-06 22:52:11 -0400131 def __init__(self, transport, protocol, loop, *, _asyncio_internal=False):
132 if not _asyncio_internal:
133 warnings.warn(f"{self.__class__} should be instaniated "
134 "by asyncio internals only, "
135 "please avoid its creation from user code",
136 DeprecationWarning)
137
Victor Stinner915bcb02014-02-01 22:49:59 +0100138 self._transport = transport
139 self._protocol = protocol
140 self._loop = loop
141 self.stdin = protocol.stdin
142 self.stdout = protocol.stdout
143 self.stderr = protocol.stderr
144 self.pid = transport.get_pid()
145
Victor Stinneracdb7822014-07-14 18:33:40 +0200146 def __repr__(self):
Yury Selivanov6370f342017-12-10 18:36:12 -0500147 return f'<{self.__class__.__name__} {self.pid}>'
Victor Stinneracdb7822014-07-14 18:33:40 +0200148
Victor Stinner915bcb02014-02-01 22:49:59 +0100149 @property
150 def returncode(self):
151 return self._transport.get_returncode()
152
Andrew Svetlov5f841b52017-12-09 00:23:48 +0200153 async def wait(self):
154 """Wait until the process exit and return the process return code."""
155 return await self._transport._wait()
Victor Stinner915bcb02014-02-01 22:49:59 +0100156
157 def send_signal(self, signal):
Victor Stinner915bcb02014-02-01 22:49:59 +0100158 self._transport.send_signal(signal)
159
160 def terminate(self):
Victor Stinner915bcb02014-02-01 22:49:59 +0100161 self._transport.terminate()
162
163 def kill(self):
Victor Stinner915bcb02014-02-01 22:49:59 +0100164 self._transport.kill()
165
Andrew Svetlov5f841b52017-12-09 00:23:48 +0200166 async def _feed_stdin(self, input):
Victor Stinnerd55b54d2014-07-17 13:12:03 +0200167 debug = self._loop.get_debug()
Victor Stinner915bcb02014-02-01 22:49:59 +0100168 self.stdin.write(input)
Victor Stinnerd55b54d2014-07-17 13:12:03 +0200169 if debug:
Yury Selivanov6370f342017-12-10 18:36:12 -0500170 logger.debug(
171 '%r communicate: feed stdin (%s bytes)', self, len(input))
Victor Stinnercc996b52014-07-17 12:25:27 +0200172 try:
Andrew Svetlov5f841b52017-12-09 00:23:48 +0200173 await self.stdin.drain()
Victor Stinnerd55b54d2014-07-17 13:12:03 +0200174 except (BrokenPipeError, ConnectionResetError) as exc:
175 # communicate() ignores BrokenPipeError and ConnectionResetError
176 if debug:
177 logger.debug('%r communicate: stdin got %r', self, exc)
Victor Stinneracdb7822014-07-14 18:33:40 +0200178
Victor Stinnerd55b54d2014-07-17 13:12:03 +0200179 if debug:
Victor Stinneracdb7822014-07-14 18:33:40 +0200180 logger.debug('%r communicate: close stdin', self)
Victor Stinner915bcb02014-02-01 22:49:59 +0100181 self.stdin.close()
182
Andrew Svetlov5f841b52017-12-09 00:23:48 +0200183 async def _noop(self):
Victor Stinner915bcb02014-02-01 22:49:59 +0100184 return None
185
Andrew Svetlov5f841b52017-12-09 00:23:48 +0200186 async def _read_stream(self, fd):
Victor Stinner915bcb02014-02-01 22:49:59 +0100187 transport = self._transport.get_pipe_transport(fd)
188 if fd == 2:
189 stream = self.stderr
190 else:
191 assert fd == 1
192 stream = self.stdout
Victor Stinneracdb7822014-07-14 18:33:40 +0200193 if self._loop.get_debug():
194 name = 'stdout' if fd == 1 else 'stderr'
195 logger.debug('%r communicate: read %s', self, name)
Andrew Svetlov5f841b52017-12-09 00:23:48 +0200196 output = await stream.read()
Victor Stinneracdb7822014-07-14 18:33:40 +0200197 if self._loop.get_debug():
198 name = 'stdout' if fd == 1 else 'stderr'
199 logger.debug('%r communicate: close %s', self, name)
Victor Stinner915bcb02014-02-01 22:49:59 +0100200 transport.close()
201 return output
202
Andrew Svetlov5f841b52017-12-09 00:23:48 +0200203 async def communicate(self, input=None):
Yury Selivanov7657f6b2016-05-13 15:35:28 -0400204 if input is not None:
Victor Stinner915bcb02014-02-01 22:49:59 +0100205 stdin = self._feed_stdin(input)
206 else:
207 stdin = self._noop()
208 if self.stdout is not None:
209 stdout = self._read_stream(1)
210 else:
211 stdout = self._noop()
212 if self.stderr is not None:
213 stderr = self._read_stream(2)
214 else:
215 stderr = self._noop()
Andrew Svetlov5f841b52017-12-09 00:23:48 +0200216 stdin, stdout, stderr = await tasks.gather(stdin, stdout, stderr,
217 loop=self._loop)
218 await self.wait()
Victor Stinner915bcb02014-02-01 22:49:59 +0100219 return (stdout, stderr)
220
221
Andrew Svetlov5f841b52017-12-09 00:23:48 +0200222async def create_subprocess_shell(cmd, stdin=None, stdout=None, stderr=None,
223 loop=None, limit=streams._DEFAULT_LIMIT,
224 **kwds):
Victor Stinner915bcb02014-02-01 22:49:59 +0100225 if loop is None:
226 loop = events.get_event_loop()
227 protocol_factory = lambda: SubprocessStreamProtocol(limit=limit,
Andrew Svetlovad4ed872019-05-06 22:52:11 -0400228 loop=loop,
229 _asyncio_internal=True)
Andrew Svetlov5f841b52017-12-09 00:23:48 +0200230 transport, protocol = await loop.subprocess_shell(
231 protocol_factory,
232 cmd, stdin=stdin, stdout=stdout,
233 stderr=stderr, **kwds)
Andrew Svetlovad4ed872019-05-06 22:52:11 -0400234 return Process(transport, protocol, loop, _asyncio_internal=True)
Victor Stinner915bcb02014-02-01 22:49:59 +0100235
Andrew Svetlov5f841b52017-12-09 00:23:48 +0200236
237async def create_subprocess_exec(program, *args, stdin=None, stdout=None,
238 stderr=None, loop=None,
239 limit=streams._DEFAULT_LIMIT, **kwds):
Victor Stinner915bcb02014-02-01 22:49:59 +0100240 if loop is None:
241 loop = events.get_event_loop()
242 protocol_factory = lambda: SubprocessStreamProtocol(limit=limit,
Andrew Svetlovad4ed872019-05-06 22:52:11 -0400243 loop=loop,
244 _asyncio_internal=True)
Andrew Svetlov5f841b52017-12-09 00:23:48 +0200245 transport, protocol = await loop.subprocess_exec(
246 protocol_factory,
247 program, *args,
248 stdin=stdin, stdout=stdout,
249 stderr=stderr, **kwds)
Andrew Svetlovad4ed872019-05-06 22:52:11 -0400250 return Process(transport, protocol, loop, _asyncio_internal=True)