gecko/testing/sisyphus/bin/timed_run.py

119 lines
3.8 KiB
Python
Executable File

#!/usr/bin/python -u
# ***** BEGIN LICENSE BLOCK *****
# Version: MPL 1.1/GPL 2.0/LGPL 2.1
#
# The contents of this file are subject to the Mozilla Public License Version
# 1.1 (the "License"); you may not use this file except in compliance with
# the License. You may obtain a copy of the License at
# http://www.mozilla.org/MPL/
#
# Software distributed under the License is distributed on an "AS IS" basis,
# WITHOUT WARRANTY OF ANY KIND, either express or implied. See the License
# for the specific language governing rights and limitations under the
# License.
#
# The Original Code is mozilla.org code.
#
# The Initial Developer of the Original Code is
# Mozilla Foundation.
# Portions created by the Initial Developer are Copyright (C) 2004
# the Initial Developer. All Rights Reserved.
#
# Contributor(s): Chris Cooper
#
# Alternatively, the contents of this file may be used under the terms of
# either the GNU General Public License Version 2 or later (the "GPL"), or
# the GNU Lesser General Public License Version 2.1 or later (the "LGPL"),
# in which case the provisions of the GPL or the LGPL are applicable instead
# of those above. If you wish to allow use of your version of this file only
# under the terms of either the GPL or the LGPL, and not to allow others to
# use your version of this file under the terms of the MPL, indicate your
# decision by deleting the provisions above and replace them with the notice
# and other provisions required by the GPL or the LGPL. If you do not delete
# the provisions above, a recipient may use your version of this file under
# the terms of any one of the MPL, the GPL or the LGPL.
#
# ***** END LICENSE BLOCK *****
# Usage: timed_run timeout prefix command args
import os, signal, sys, time
#
# returns exit code as follows:
#
exitOSError = 66
exitSignal = 77
exitTimeout = 88
exitInterrupt = 99
pid = None
prefix = sys.argv[2]
elapsedtime = 0
if prefix == "-":
prefix = ''
else:
prefix = prefix + ':'
def alarm_handler(signum, frame):
global pid
global prefix
try:
print "%s EXIT STATUS: TIMED OUT (%s seconds)" % (prefix, sys.argv[1])
sys.stdout.flush()
os.kill(pid, signal.SIGKILL)
except OSError, e:
print "timed_run.py: exception trying to kill process: %d (%s)" % (e.errno, e.strerror)
pass
sys.exit(exitTimeout)
def forkexec(command, args):
global prefix
global elapsedtime
#print command
#print args
try:
pid = os.fork()
if pid == 0: # Child
os.execvp(command, args)
else: # Parent
return pid
except OSError, e:
print "%s ERROR: %s %s failed: %d (%s) (%f seconds)" % (prefix, command, args, e.errno, e.strerror, elapsedtime)
sys.exit(exitOSError)
signal.signal(signal.SIGALRM, alarm_handler)
signal.alarm(int(sys.argv[1]))
starttime = time.time()
try:
pid = forkexec(sys.argv[3], sys.argv[3:])
status = os.waitpid(pid, 0)[1]
signal.alarm(0) # Cancel the alarm
stoptime = time.time()
elapsedtime = stoptime - starttime
# it appears that linux at least will on "occasion" return a status
# when the process was terminated by a signal, so test signal first.
if os.WIFSIGNALED(status):
print "%s EXIT STATUS: CRASHED signal %d (%f seconds)" % (prefix, os.WTERMSIG(status), elapsedtime)
sys.exit(exitSignal)
elif os.WIFEXITED(status):
rc = os.WEXITSTATUS(status)
msg = ''
if rc == 0:
msg = 'NORMAL'
elif rc < 3:
msg = 'ABNORMAL ' + str(rc)
rc = exitSignal
else:
msg = 'CRASHED ' + str(rc)
rc = exitSignal
print "%s EXIT STATUS: %s (%f seconds)" % (prefix, msg, elapsedtime)
sys.exit(rc)
else:
print "%s EXIT STATUS: NONE (%f seconds)" % (prefix, elapsedtime)
sys.exit(0)
except KeyboardInterrupt:
os.kill(pid, 9)
sys.exit(exitInterrupt)