 Chromium Code Reviews
 Chromium Code Reviews Issue 14247012:
  Add support for parallel presubmit unit testing.  (Closed) 
  Base URL: https://chromium.googlesource.com/chromium/tools/depot_tools.git@master
    
  
    Issue 14247012:
  Add support for parallel presubmit unit testing.  (Closed) 
  Base URL: https://chromium.googlesource.com/chromium/tools/depot_tools.git@master| Index: presubmit_support.py | 
| diff --git a/presubmit_support.py b/presubmit_support.py | 
| index fe1c2878c5c27b64e75dfd36e2b456ac84aa43af..98b065925bec49db82f45c81c13e6e18c1a71fc9 100755 | 
| --- a/presubmit_support.py | 
| +++ b/presubmit_support.py | 
| @@ -15,6 +15,7 @@ __version__ = '1.6.2' | 
| import cpplint | 
| import cPickle # Exposed through the API. | 
| import cStringIO # Exposed through the API. | 
| +import collections | 
| import contextlib | 
| import fnmatch | 
| import glob | 
| @@ -22,6 +23,7 @@ import inspect | 
| import json # Exposed through the API. | 
| import logging | 
| import marshal # Exposed through the API. | 
| +import multiprocessing | 
| import optparse | 
| import os # Somewhat exposed through the API. | 
| import pickle # Exposed through the API. | 
| @@ -54,6 +56,9 @@ class PresubmitFailure(Exception): | 
| pass | 
| +CommandData = collections.namedtuple('CommandData', | 
| + ['name', 'cmd', 'kwargs', 'message']) | 
| + | 
| def normpath(path): | 
| '''Version of os.path.normpath that also changes backward slashes to | 
| forward slashes when not running on Windows. | 
| @@ -104,81 +109,89 @@ class PresubmitOutput(object): | 
| return ''.join(self.written_output) | 
| +class _PresubmitResult(object): | 
| + """Base class for result objects.""" | 
| + fatal = False | 
| + should_prompt = False | 
| + | 
| + def __init__(self, message, items=None, long_text=''): | 
| + """ | 
| + message: A short one-line message to indicate errors. | 
| + items: A list of short strings to indicate where errors occurred. | 
| + long_text: multi-line text output, e.g. from another tool | 
| + """ | 
| + self._message = message | 
| + self._items = [] | 
| + if items: | 
| + self._items = items | 
| + self._long_text = long_text.rstrip() | 
| + | 
| + def handle(self, output): | 
| + output.write(self._message) | 
| + output.write('\n') | 
| + for index, item in enumerate(self._items): | 
| + output.write(' ') | 
| + # Write separately in case it's unicode. | 
| + output.write(str(item)) | 
| + if index < len(self._items) - 1: | 
| + output.write(' \\') | 
| + output.write('\n') | 
| + if self._long_text: | 
| + output.write('\n***************\n') | 
| + # Write separately in case it's unicode. | 
| + output.write(self._long_text) | 
| + output.write('\n***************\n') | 
| + if self.fatal: | 
| + output.fail() | 
| + | 
| 
M-A Ruel
2013/04/19 16:49:39
two lines between file level symbols
 
Isaac (away)
2013/04/20 00:41:36
Done.
 | 
| +class _PresubmitAddReviewers(_PresubmitResult): | 
| + """Add some suggested reviewers to the change.""" | 
| + def __init__(self, reviewers): | 
| + super(OutputApi.PresubmitAddReviewers, self).__init__('') | 
| 
M-A Ruel
2013/04/19 16:49:39
Stale reference to OutputApi (and below)
 
Isaac (away)
2013/04/20 00:41:36
Done.
 | 
| + self.reviewers = reviewers | 
| + | 
| + def handle(self, output): | 
| + output.reviewers.extend(self.reviewers) | 
| + | 
| +class _PresubmitError(_PresubmitResult): | 
| + """A hard presubmit error.""" | 
| + fatal = True | 
| + | 
| +class _PresubmitPromptWarning(_PresubmitResult): | 
| + """An warning that prompts the user if they want to continue.""" | 
| + should_prompt = True | 
| + | 
| +class _PresubmitNotifyResult(_PresubmitResult): | 
| + """Just print something to the screen -- but it's not even a warning.""" | 
| + pass | 
| + | 
| +class _MailTextResult(_PresubmitResult): | 
| + """A warning that should be included in the review request email.""" | 
| + def __init__(self, *args, **kwargs): | 
| + super(OutputApi.MailTextResult, self).__init__() | 
| + raise NotImplementedError() | 
| + | 
| class OutputApi(object): | 
| """An instance of OutputApi gets passed to presubmit scripts so that they | 
| can output various types of results. | 
| """ | 
| + # multiprocessing requires these be top level modules | 
| + PresubmitResult = _PresubmitResult | 
| + PresubmitAddReviewers = _PresubmitAddReviewers | 
| + PresubmitError = _PresubmitError | 
| + PresubmitPromptWarning = _PresubmitPromptWarning | 
| + PresubmitNotifyResult = _PresubmitNotifyResult | 
| + MailTextResult = _MailTextResult | 
| + | 
| def __init__(self, is_committing): | 
| self.is_committing = is_committing | 
| - class PresubmitResult(object): | 
| - """Base class for result objects.""" | 
| - fatal = False | 
| - should_prompt = False | 
| - | 
| - def __init__(self, message, items=None, long_text=''): | 
| - """ | 
| - message: A short one-line message to indicate errors. | 
| - items: A list of short strings to indicate where errors occurred. | 
| - long_text: multi-line text output, e.g. from another tool | 
| - """ | 
| - self._message = message | 
| - self._items = [] | 
| - if items: | 
| - self._items = items | 
| - self._long_text = long_text.rstrip() | 
| - | 
| - def handle(self, output): | 
| - output.write(self._message) | 
| - output.write('\n') | 
| - for index, item in enumerate(self._items): | 
| - output.write(' ') | 
| - # Write separately in case it's unicode. | 
| - output.write(str(item)) | 
| - if index < len(self._items) - 1: | 
| - output.write(' \\') | 
| - output.write('\n') | 
| - if self._long_text: | 
| - output.write('\n***************\n') | 
| - # Write separately in case it's unicode. | 
| - output.write(self._long_text) | 
| - output.write('\n***************\n') | 
| - if self.fatal: | 
| - output.fail() | 
| - | 
| - class PresubmitAddReviewers(PresubmitResult): | 
| - """Add some suggested reviewers to the change.""" | 
| - def __init__(self, reviewers): | 
| - super(OutputApi.PresubmitAddReviewers, self).__init__('') | 
| - self.reviewers = reviewers | 
| - | 
| - def handle(self, output): | 
| - output.reviewers.extend(self.reviewers) | 
| - | 
| - class PresubmitError(PresubmitResult): | 
| - """A hard presubmit error.""" | 
| - fatal = True | 
| - | 
| - class PresubmitPromptWarning(PresubmitResult): | 
| - """An warning that prompts the user if they want to continue.""" | 
| - should_prompt = True | 
| - | 
| - class PresubmitNotifyResult(PresubmitResult): | 
| - """Just print something to the screen -- but it's not even a warning.""" | 
| - pass | 
| - | 
| def PresubmitPromptOrNotify(self, *args, **kwargs): | 
| """Warn the user when uploading, but only notify if committing.""" | 
| if self.is_committing: | 
| return self.PresubmitNotifyResult(*args, **kwargs) | 
| return self.PresubmitPromptWarning(*args, **kwargs) | 
| - class MailTextResult(PresubmitResult): | 
| - """A warning that should be included in the review request email.""" | 
| - def __init__(self, *args, **kwargs): | 
| - super(OutputApi.MailTextResult, self).__init__() | 
| - raise NotImplementedError() | 
| - | 
| class InputApi(object): | 
| """An instance of this object is passed to presubmit scripts so they can | 
| @@ -284,6 +297,7 @@ class InputApi(object): | 
| self.owners_db = owners.Database(change.RepositoryRoot(), | 
| fopen=file, os_path=self.os_path, glob=self.glob) | 
| self.verbose = verbose | 
| + self.Command = CommandData | 
| # Replace <hash_map> and <hash_set> as headers that need to be included | 
| # with "base/hash_tables.h" instead. | 
| @@ -437,6 +451,26 @@ class InputApi(object): | 
| """Returns if a change is TBR'ed.""" | 
| return 'TBR' in self.change.tags | 
| + def RunTests(self, tests_mix, parallel=True): | 
| 
M-A Ruel
2013/04/19 16:49:39
Will parallel=False ever make sense?
 
Isaac (away)
2013/04/19 19:52:04
Yes, it is used by the legacy commands in presubmi
 | 
| + tests = [] | 
| + msgs = [] | 
| + for t in tests_mix: | 
| + if isinstance(t, OutputApi.PresubmitResult): | 
| + msgs.append(t) | 
| + else: | 
| + t = t._replace(kwargs=t.kwargs.copy()) | 
| + t.kwargs.setdefault('env', self.environ) | 
| + t.kwargs.setdefault('cwd', self.PresubmitLocalPath()) | 
| + tests.append(t) | 
| + if parallel: | 
| + pool = multiprocessing.Pool() | 
| + msgs.extend(pool.map_async(CallCommand, tests).get(99999)) | 
| 
M-A Ruel
2013/04/19 16:49:39
Can you note why this is used? (e.g. the Ctrl-C ha
 
Isaac (away)
2013/04/20 00:41:36
Done.
 | 
| + pool.close() | 
| + pool.join() | 
| + else: | 
| + msgs.extend(map(CallCommand, tests)) | 
| + return [m for m in msgs if m is not None] | 
| 
M-A Ruel
2013/04/19 16:49:39
return [m for m in msgs if m]
or
return filter(Non
 
Isaac (away)
2013/04/19 19:52:04
The intention is that each item in msg list is eit
 | 
| + | 
| class AffectedFile(object): | 
| """Representation of a file in a change.""" | 
| @@ -1238,6 +1272,19 @@ def canned_check_filter(method_names): | 
| for name, method in filtered.iteritems(): | 
| setattr(presubmit_canned_checks, name, method) | 
| +# multiprocessing requires a top level fun with a single argument | 
| 
M-A Ruel
2013/04/19 16:49:39
fun -> function
I'd prefer the code to be inside t
 
Isaac (away)
2013/04/20 00:41:36
Done.
 | 
| +def CallCommand(cmd_data): | 
| + cmd_data.kwargs['stdout'] = subprocess.PIPE | 
| + cmd_data.kwargs['stderr'] = subprocess.STDOUT | 
| + try: | 
| + (out, _), code = subprocess.communicate(cmd_data.cmd, **cmd_data.kwargs) | 
| + if code != 0: | 
| + #import pdb; pdb.set_trace() | 
| 
M-A Ruel
2013/04/19 16:49:39
Remove before committing
 
Isaac (away)
2013/04/20 00:41:36
Done.
 | 
| + return cmd_data.message('%s failed\n%s' % (cmd_data.name, out)) | 
| + except OSError as e: | 
| + return cmd_data.message( | 
| + '%s exec failure\n %s\n%s' % (cmd_data.name, e, out)) | 
| + | 
| def Main(argv): | 
| parser = optparse.OptionParser(usage="%prog [options] <files...>", |