# Copyright (c) 2009, Google Inc. All rights reserved. # Copyright (c) 2009 Apple Inc. All rights reserved. # # Redistribution and use in source and binary forms, with or without # modification, are permitted provided that the following conditions are # met: # # * Redistributions of source code must retain the above copyright # notice, this list of conditions and the following disclaimer. # * Redistributions in binary form must reproduce the above # copyright notice, this list of conditions and the following disclaimer # in the documentation and/or other materials provided with the # distribution. # * Neither the name of Google Inc. nor the names of its # contributors may be used to endorse or promote products derived from # this software without specific prior written permission. # # THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS # "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT # LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR # A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT # OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, # SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT # LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, # DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY # THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT # (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE # OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. import os import StringIO import subprocess import sys from webkitpy.webkit_logging import tee class ScriptError(Exception): def __init__(self, message=None, script_args=None, exit_code=None, output=None, cwd=None): if not message: message = 'Failed to run "%s"' % script_args if exit_code: message += " exit_code: %d" % exit_code if cwd: message += " cwd: %s" % cwd Exception.__init__(self, message) self.script_args = script_args # 'args' is already used by Exception self.exit_code = exit_code self.output = output self.cwd = cwd def message_with_output(self, output_limit=500): if self.output: if output_limit and len(self.output) > output_limit: return "%s\nLast %s characters of output:\n%s" % \ (self, output_limit, self.output[-output_limit:]) return "%s\n%s" % (self, self.output) return str(self) def command_name(self): command_path = self.script_args if type(command_path) is list: command_path = command_path[0] return os.path.basename(command_path) def run_command(*args, **kwargs): # FIXME: This should not be a global static. # New code should use Executive.run_command directly instead return Executive().run_command(*args, **kwargs) class Executive(object): def _run_command_with_teed_output(self, args, teed_output): child_process = subprocess.Popen(args, stdout=subprocess.PIPE, stderr=subprocess.STDOUT) # Use our own custom wait loop because Popen ignores a tee'd # stderr/stdout. # FIXME: This could be improved not to flatten output to stdout. while True: output_line = child_process.stdout.readline() if output_line == "" and child_process.poll() != None: return child_process.poll() teed_output.write(output_line) def run_and_throw_if_fail(self, args, quiet=False): # Cache the child's output locally so it can be used for error reports. child_out_file = StringIO.StringIO() if quiet: dev_null = open(os.devnull, "w") child_stdout = tee(child_out_file, dev_null if quiet else sys.stdout) exit_code = self._run_command_with_teed_output(args, child_stdout) if quiet: dev_null.close() child_output = child_out_file.getvalue() child_out_file.close() if exit_code: raise ScriptError(script_args=args, exit_code=exit_code, output=child_output) @staticmethod def cpu_count(): # This API exists only in Python 2.6 and higher. :( try: import multiprocessing return multiprocessing.cpu_count() except (ImportError, NotImplementedError): # This quantity is a lie but probably a reasonable guess for modern # machines. return 2 # Error handlers do not need to be static methods once all callers are # updated to use an Executive object. @staticmethod def default_error_handler(error): raise error @staticmethod def ignore_error(error): pass # FIXME: This should be merged with run_and_throw_if_fail def run_command(self, args, cwd=None, input=None, error_handler=None, return_exit_code=False, return_stderr=True): if hasattr(input, 'read'): # Check if the input is a file. stdin = input string_to_communicate = None else: stdin = subprocess.PIPE if input else None string_to_communicate = input if return_stderr: stderr = subprocess.STDOUT else: stderr = None process = subprocess.Popen(args, stdin=stdin, stdout=subprocess.PIPE, stderr=stderr, cwd=cwd) output = process.communicate(string_to_communicate)[0] exit_code = process.wait() if exit_code: script_error = ScriptError(script_args=args, exit_code=exit_code, output=output, cwd=cwd) (error_handler or self.default_error_handler)(script_error) if return_exit_code: return exit_code return output