diff options
author | phajdan.jr@chromium.org <phajdan.jr@chromium.org@0039d316-1c4b-4281-b951-d872f2087c98> | 2010-09-07 20:05:56 +0000 |
---|---|---|
committer | phajdan.jr@chromium.org <phajdan.jr@chromium.org@0039d316-1c4b-4281-b951-d872f2087c98> | 2010-09-07 20:05:56 +0000 |
commit | a65b3700d590f84cfb9bbc93b652a0f3bb54893a (patch) | |
tree | de65bbc2152a07837cb6c4f922aba5054206f684 /tools | |
parent | 90c7dae0e0d58deafb29fab338f96c67661d512f (diff) | |
download | chromium_src-a65b3700d590f84cfb9bbc93b652a0f3bb54893a.zip chromium_src-a65b3700d590f84cfb9bbc93b652a0f3bb54893a.tar.gz chromium_src-a65b3700d590f84cfb9bbc93b652a0f3bb54893a.tar.bz2 |
Make parallel test launcher print output as it's written by the child.
Add protection against hang when the child leaks a process.
BUG=54098
TEST=none
Review URL: http://codereview.chromium.org/3360008
git-svn-id: svn://svn.chromium.org/chrome/trunk/src@58741 0039d316-1c4b-4281-b951-d872f2087c98
Diffstat (limited to 'tools')
-rwxr-xr-x | tools/parallel_launcher/parallel_launcher.py | 69 |
1 files changed, 61 insertions, 8 deletions
diff --git a/tools/parallel_launcher/parallel_launcher.py b/tools/parallel_launcher/parallel_launcher.py index a29e05b..9a8e6ab 100755 --- a/tools/parallel_launcher/parallel_launcher.py +++ b/tools/parallel_launcher/parallel_launcher.py @@ -16,7 +16,48 @@ import optparse import os import subprocess import sys -import tempfile +import threading +import time + + +def StreamCopyWindows(stream_from, stream_to): + """Copies stream_from to stream_to.""" + + while True: + buf = stream_from.read(1024) + if not buf: + break + stream_to.write(buf) + stream_to.flush() + +def StreamCopyPosix(stream_from, stream_to, child_exited): + """ + Copies stream_from to stream_to, and exits if child_exited + is signaled. + """ + + import fcntl + + # Put the source stream in a non-blocking mode, so we can check + # child_exited when there is no data. + fd = stream_from.fileno() + fl = fcntl.fcntl(fd, fcntl.F_GETFL) + fcntl.fcntl(fd, fcntl.F_SETFL, fl | os.O_NONBLOCK) + + while True: + try: + buf = os.read(fd, 1024) + except OSError, e: + if e.errno == 11: + if child_exited.isSet(): + break + time.sleep(0.1) + continue + raise + if not buf: + break + stream_to.write(buf) + stream_to.flush() class TestLauncher(object): def __init__(self, args, executable, num_shards, shard): @@ -25,7 +66,6 @@ class TestLauncher(object): self._num_shards = num_shards self._shard = shard self._test = None - self._tempfile = tempfile.TemporaryFile() def launch(self): env = os.environ.copy() @@ -33,16 +73,29 @@ class TestLauncher(object): env['GTEST_SHARD_INDEX'] = str(self._shard) self._test = subprocess.Popen(args=self._args, executable=self._executable, - stdout=self._tempfile, + stdout=subprocess.PIPE, stderr=subprocess.STDOUT, env=env) def wait(self): - code = self._test.wait() - self._tempfile.seek(0) - print self._tempfile.read() - self._tempfile.close() - return code + if subprocess.mswindows: + stdout_thread = threading.Thread( + target=StreamCopyWindows, + args=[self._test.stdout, sys.stdout]) + stdout_thread.start() + code = self._test.wait() + stdout_thread.join() + return code + else: + child_exited = threading.Event() + stdout_thread = threading.Thread( + target=StreamCopyPosix, + args=[self._test.stdout, sys.stdout, child_exited]) + stdout_thread.start() + code = self._test.wait() + child_exited.set() + stdout_thread.join() + return code def main(argv): parser = optparse.OptionParser() |