blob: 4c85466859f8f0f195d5fd0fc72c269b7cdca2b8 [file] [log] [blame]
Victor Stinner915bcb02014-02-01 22:49:59 +01001__all__ = ['create_subprocess_exec', 'create_subprocess_shell']
2
Victor Stinner915bcb02014-02-01 22:49:59 +01003import subprocess
4
5from . import events
Victor Stinner915bcb02014-02-01 22:49:59 +01006from . import protocols
7from . import streams
8from . import tasks
Victor Stinnerf951d282014-06-29 00:46:45 +02009from .coroutines import coroutine
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
22 def __init__(self, limit, loop):
23 super().__init__(loop=loop)
24 self._limit = limit
25 self.stdin = self.stdout = self.stderr = None
Victor Stinner915bcb02014-02-01 22:49:59 +010026 self._transport = None
Seth M. Larson604faba2017-03-02 22:21:18 -060027 self._process_exited = False
28 self._pipe_fds = []
Victor Stinner915bcb02014-02-01 22:49:59 +010029
Victor Stinneracdb7822014-07-14 18:33:40 +020030 def __repr__(self):
31 info = [self.__class__.__name__]
32 if self.stdin is not None:
33 info.append('stdin=%r' % self.stdin)
34 if self.stdout is not None:
35 info.append('stdout=%r' % self.stdout)
36 if self.stderr is not None:
37 info.append('stderr=%r' % self.stderr)
38 return '<%s>' % ' '.join(info)
39
Victor Stinner915bcb02014-02-01 22:49:59 +010040 def connection_made(self, transport):
41 self._transport = transport
Victor Stinner5ef586f2014-11-25 17:20:33 +010042
43 stdout_transport = transport.get_pipe_transport(1)
44 if stdout_transport is not None:
Victor Stinner915bcb02014-02-01 22:49:59 +010045 self.stdout = streams.StreamReader(limit=self._limit,
46 loop=self._loop)
Victor Stinner5ef586f2014-11-25 17:20:33 +010047 self.stdout.set_transport(stdout_transport)
Seth M. Larson604faba2017-03-02 22:21:18 -060048 self._pipe_fds.append(1)
Victor Stinner5ef586f2014-11-25 17:20:33 +010049
50 stderr_transport = transport.get_pipe_transport(2)
51 if stderr_transport is not None:
Victor Stinner915bcb02014-02-01 22:49:59 +010052 self.stderr = streams.StreamReader(limit=self._limit,
53 loop=self._loop)
Victor Stinner5ef586f2014-11-25 17:20:33 +010054 self.stderr.set_transport(stderr_transport)
Seth M. Larson604faba2017-03-02 22:21:18 -060055 self._pipe_fds.append(2)
Victor Stinner5ef586f2014-11-25 17:20:33 +010056
57 stdin_transport = transport.get_pipe_transport(0)
58 if stdin_transport is not None:
59 self.stdin = streams.StreamWriter(stdin_transport,
Victor Stinner915bcb02014-02-01 22:49:59 +010060 protocol=self,
61 reader=None,
62 loop=self._loop)
Victor Stinnerf651a602015-01-14 02:10:33 +010063
Victor Stinner915bcb02014-02-01 22:49:59 +010064 def pipe_data_received(self, fd, data):
65 if fd == 1:
66 reader = self.stdout
67 elif fd == 2:
68 reader = self.stderr
69 else:
70 reader = None
71 if reader is not None:
72 reader.feed_data(data)
73
74 def pipe_connection_lost(self, fd, exc):
75 if fd == 0:
76 pipe = self.stdin
77 if pipe is not None:
78 pipe.close()
79 self.connection_lost(exc)
80 return
81 if fd == 1:
82 reader = self.stdout
83 elif fd == 2:
84 reader = self.stderr
85 else:
86 reader = None
87 if reader != None:
88 if exc is None:
89 reader.feed_eof()
90 else:
91 reader.set_exception(exc)
92
Seth M. Larson604faba2017-03-02 22:21:18 -060093 if fd in self._pipe_fds:
94 self._pipe_fds.remove(fd)
95 self._maybe_close_transport()
96
Victor Stinner915bcb02014-02-01 22:49:59 +010097 def process_exited(self):
Seth M. Larson604faba2017-03-02 22:21:18 -060098 self._process_exited = True
99 self._maybe_close_transport()
100
101 def _maybe_close_transport(self):
102 if len(self._pipe_fds) == 0 and self._process_exited:
103 self._transport.close()
104 self._transport = None
Victor Stinner791009b2015-01-15 13:16:02 +0100105
Victor Stinner915bcb02014-02-01 22:49:59 +0100106
107class Process:
108 def __init__(self, transport, protocol, loop):
109 self._transport = transport
110 self._protocol = protocol
111 self._loop = loop
112 self.stdin = protocol.stdin
113 self.stdout = protocol.stdout
114 self.stderr = protocol.stderr
115 self.pid = transport.get_pid()
116
Victor Stinneracdb7822014-07-14 18:33:40 +0200117 def __repr__(self):
118 return '<%s %s>' % (self.__class__.__name__, self.pid)
119
Victor Stinner915bcb02014-02-01 22:49:59 +0100120 @property
121 def returncode(self):
122 return self._transport.get_returncode()
123
Victor Stinnerf951d282014-06-29 00:46:45 +0200124 @coroutine
Victor Stinner915bcb02014-02-01 22:49:59 +0100125 def wait(self):
Victor Stinner47cd10d2015-01-30 00:05:19 +0100126 """Wait until the process exit and return the process return code.
Victor Stinner915bcb02014-02-01 22:49:59 +0100127
Victor Stinner47cd10d2015-01-30 00:05:19 +0100128 This method is a coroutine."""
Victor Stinner1241ecc2015-01-30 00:16:14 +0100129 return (yield from self._transport._wait())
Victor Stinner915bcb02014-02-01 22:49:59 +0100130
131 def send_signal(self, signal):
Victor Stinner915bcb02014-02-01 22:49:59 +0100132 self._transport.send_signal(signal)
133
134 def terminate(self):
Victor Stinner915bcb02014-02-01 22:49:59 +0100135 self._transport.terminate()
136
137 def kill(self):
Victor Stinner915bcb02014-02-01 22:49:59 +0100138 self._transport.kill()
139
Victor Stinnerf951d282014-06-29 00:46:45 +0200140 @coroutine
Victor Stinner915bcb02014-02-01 22:49:59 +0100141 def _feed_stdin(self, input):
Victor Stinnerd55b54d2014-07-17 13:12:03 +0200142 debug = self._loop.get_debug()
Victor Stinner915bcb02014-02-01 22:49:59 +0100143 self.stdin.write(input)
Victor Stinnerd55b54d2014-07-17 13:12:03 +0200144 if debug:
Victor Stinneracdb7822014-07-14 18:33:40 +0200145 logger.debug('%r communicate: feed stdin (%s bytes)',
146 self, len(input))
Victor Stinnercc996b52014-07-17 12:25:27 +0200147 try:
148 yield from self.stdin.drain()
Victor Stinnerd55b54d2014-07-17 13:12:03 +0200149 except (BrokenPipeError, ConnectionResetError) as exc:
150 # communicate() ignores BrokenPipeError and ConnectionResetError
151 if debug:
152 logger.debug('%r communicate: stdin got %r', self, exc)
Victor Stinneracdb7822014-07-14 18:33:40 +0200153
Victor Stinnerd55b54d2014-07-17 13:12:03 +0200154 if debug:
Victor Stinneracdb7822014-07-14 18:33:40 +0200155 logger.debug('%r communicate: close stdin', self)
Victor Stinner915bcb02014-02-01 22:49:59 +0100156 self.stdin.close()
157
Victor Stinnerf951d282014-06-29 00:46:45 +0200158 @coroutine
Victor Stinner915bcb02014-02-01 22:49:59 +0100159 def _noop(self):
160 return None
161
Victor Stinnerf951d282014-06-29 00:46:45 +0200162 @coroutine
Victor Stinner915bcb02014-02-01 22:49:59 +0100163 def _read_stream(self, fd):
164 transport = self._transport.get_pipe_transport(fd)
165 if fd == 2:
166 stream = self.stderr
167 else:
168 assert fd == 1
169 stream = self.stdout
Victor Stinneracdb7822014-07-14 18:33:40 +0200170 if self._loop.get_debug():
171 name = 'stdout' if fd == 1 else 'stderr'
172 logger.debug('%r communicate: read %s', self, name)
Victor Stinner915bcb02014-02-01 22:49:59 +0100173 output = yield from stream.read()
Victor Stinneracdb7822014-07-14 18:33:40 +0200174 if self._loop.get_debug():
175 name = 'stdout' if fd == 1 else 'stderr'
176 logger.debug('%r communicate: close %s', self, name)
Victor Stinner915bcb02014-02-01 22:49:59 +0100177 transport.close()
178 return output
179
Victor Stinnerf951d282014-06-29 00:46:45 +0200180 @coroutine
Victor Stinner915bcb02014-02-01 22:49:59 +0100181 def communicate(self, input=None):
Yury Selivanov7657f6b2016-05-13 15:35:28 -0400182 if input is not None:
Victor Stinner915bcb02014-02-01 22:49:59 +0100183 stdin = self._feed_stdin(input)
184 else:
185 stdin = self._noop()
186 if self.stdout is not None:
187 stdout = self._read_stream(1)
188 else:
189 stdout = self._noop()
190 if self.stderr is not None:
191 stderr = self._read_stream(2)
192 else:
193 stderr = self._noop()
194 stdin, stdout, stderr = yield from tasks.gather(stdin, stdout, stderr,
Victor Stinner8d0230b2014-02-20 10:12:59 +0100195 loop=self._loop)
Victor Stinner915bcb02014-02-01 22:49:59 +0100196 yield from self.wait()
197 return (stdout, stderr)
198
199
Victor Stinnerf951d282014-06-29 00:46:45 +0200200@coroutine
Victor Stinner915bcb02014-02-01 22:49:59 +0100201def create_subprocess_shell(cmd, stdin=None, stdout=None, stderr=None,
202 loop=None, limit=streams._DEFAULT_LIMIT, **kwds):
203 if loop is None:
204 loop = events.get_event_loop()
205 protocol_factory = lambda: SubprocessStreamProtocol(limit=limit,
206 loop=loop)
207 transport, protocol = yield from loop.subprocess_shell(
208 protocol_factory,
209 cmd, stdin=stdin, stdout=stdout,
210 stderr=stderr, **kwds)
Victor Stinner915bcb02014-02-01 22:49:59 +0100211 return Process(transport, protocol, loop)
212
Victor Stinnerf951d282014-06-29 00:46:45 +0200213@coroutine
Yury Selivanov57797522014-02-18 22:56:15 -0500214def create_subprocess_exec(program, *args, stdin=None, stdout=None,
215 stderr=None, loop=None,
216 limit=streams._DEFAULT_LIMIT, **kwds):
Victor Stinner915bcb02014-02-01 22:49:59 +0100217 if loop is None:
218 loop = events.get_event_loop()
219 protocol_factory = lambda: SubprocessStreamProtocol(limit=limit,
220 loop=loop)
221 transport, protocol = yield from loop.subprocess_exec(
222 protocol_factory,
Victor Stinner20e07432014-02-11 11:44:56 +0100223 program, *args,
224 stdin=stdin, stdout=stdout,
Victor Stinner915bcb02014-02-01 22:49:59 +0100225 stderr=stderr, **kwds)
Victor Stinner915bcb02014-02-01 22:49:59 +0100226 return Process(transport, protocol, loop)