| # Copyright 2016 the V8 project authors. All rights reserved. |
| # Use of this source code is governed by a BSD-style license that can be |
| # found in the LICENSE file. |
| |
| # Fork from commands.py and output.py in v8 test driver. |
| |
| import signal |
| import subprocess |
| import sys |
| from threading import Event, Timer |
| |
| |
| class Output(object): |
| def __init__(self, exit_code, timed_out, stdout, pid): |
| self.exit_code = exit_code |
| self.timed_out = timed_out |
| self.stdout = stdout |
| self.pid = pid |
| |
| def HasCrashed(self): |
| # Timed out tests will have exit_code -signal.SIGTERM. |
| if self.timed_out: |
| return False |
| return (self.exit_code < 0 and |
| self.exit_code != -signal.SIGABRT) |
| |
| def HasTimedOut(self): |
| return self.timed_out |
| |
| |
| def Execute(args, cwd, timeout=None): |
| popen_args = [c for c in args if c != ""] |
| try: |
| process = subprocess.Popen( |
| args=popen_args, |
| stdout=subprocess.PIPE, |
| stderr=subprocess.STDOUT, |
| cwd=cwd |
| ) |
| except Exception as e: |
| sys.stderr.write("Error executing: %s\n" % popen_args) |
| raise e |
| |
| timeout_event = Event() |
| |
| def kill_process(): |
| timeout_event.set() |
| try: |
| process.kill() |
| except OSError: |
| sys.stderr.write('Error: Process %s already ended.\n' % process.pid) |
| |
| |
| timer = Timer(timeout, kill_process) |
| timer.start() |
| stdout, _ = process.communicate() |
| timer.cancel() |
| |
| return Output( |
| process.returncode, |
| timeout_event.is_set(), |
| stdout.decode('utf-8', 'replace').encode('utf-8'), |
| process.pid, |
| ) |