एक बार जब आप * यूनिक्स में फुल प्रोसेस रनिंग मशीनरी को समझ लेते हैं, तो आपको आसानी से सरल समाधान मिल जाएगा:
इस सरल उदाहरण पर विचार करें कि टाइमआउटेबल कम्युनिकेशन () मेथ का चयन select.select () (आजकल उपलब्ध हर अलथेरे * पर उपलब्ध) का उपयोग कैसे करें। यह भी epoll / poll / kqueue के साथ लिखा जा सकता है, लेकिन select.select () संस्करण आपके लिए एक अच्छा उदाहरण हो सकता है। और आपके कार्य के लिए select.select () (गति और 1024 अधिकतम fds) की प्रमुख सीमाएँ लागू नहीं होती हैं।
यह * निक्स के तहत काम करता है, थ्रेड्स नहीं बनाता है, सिग्नल का उपयोग नहीं करता है, किसी भी धागे (केवल मुख्य नहीं) से प्रशंसित किया जा सकता है, और मेरी मशीन (i5 2.3ghz) पर stdout से डेटा के 250mb / s पढ़ने के लिए पर्याप्त पर्याप्त है।
संचार के अंत में joinding stdout / stderr में एक समस्या है। यदि आपके पास बहुत बड़ा प्रोग्राम आउटपुट है तो इससे बड़ी मेमोरी उपयोग हो सकती है। लेकिन आप छोटे टाइमआउट के साथ कई बार कम्यूनिकेशन () कॉल कर सकते हैं।
class Popen(subprocess.Popen):
def communicate(self, input=None, timeout=None):
if timeout is None:
return subprocess.Popen.communicate(self, input)
if self.stdin:
# Flush stdio buffer, this might block if user
# has been writing to .stdin in an uncontrolled
# fashion.
self.stdin.flush()
if not input:
self.stdin.close()
read_set, write_set = [], []
stdout = stderr = None
if self.stdin and input:
write_set.append(self.stdin)
if self.stdout:
read_set.append(self.stdout)
stdout = []
if self.stderr:
read_set.append(self.stderr)
stderr = []
input_offset = 0
deadline = time.time() + timeout
while read_set or write_set:
try:
rlist, wlist, xlist = select.select(read_set, write_set, [], max(0, deadline - time.time()))
except select.error as ex:
if ex.args[0] == errno.EINTR:
continue
raise
if not (rlist or wlist):
# Just break if timeout
# Since we do not close stdout/stderr/stdin, we can call
# communicate() several times reading data by smaller pieces.
break
if self.stdin in wlist:
chunk = input[input_offset:input_offset + subprocess._PIPE_BUF]
try:
bytes_written = os.write(self.stdin.fileno(), chunk)
except OSError as ex:
if ex.errno == errno.EPIPE:
self.stdin.close()
write_set.remove(self.stdin)
else:
raise
else:
input_offset += bytes_written
if input_offset >= len(input):
self.stdin.close()
write_set.remove(self.stdin)
# Read stdout / stderr by 1024 bytes
for fn, tgt in (
(self.stdout, stdout),
(self.stderr, stderr),
):
if fn in rlist:
data = os.read(fn.fileno(), 1024)
if data == '':
fn.close()
read_set.remove(fn)
tgt.append(data)
if stdout is not None:
stdout = ''.join(stdout)
if stderr is not None:
stderr = ''.join(stderr)
return (stdout, stderr)