summaryrefslogtreecommitdiffstats
path: root/tools/parallel_launcher
diff options
context:
space:
mode:
authorphajdan.jr@chromium.org <phajdan.jr@chromium.org@0039d316-1c4b-4281-b951-d872f2087c98>2010-09-07 20:05:56 +0000
committerphajdan.jr@chromium.org <phajdan.jr@chromium.org@0039d316-1c4b-4281-b951-d872f2087c98>2010-09-07 20:05:56 +0000
commita65b3700d590f84cfb9bbc93b652a0f3bb54893a (patch)
treede65bbc2152a07837cb6c4f922aba5054206f684 /tools/parallel_launcher
parent90c7dae0e0d58deafb29fab338f96c67661d512f (diff)
downloadchromium_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/parallel_launcher')
-rwxr-xr-xtools/parallel_launcher/parallel_launcher.py69
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()