mirror of https://github.com/ARMmbed/mbed-os.git
Merge pull request #3922 from adbridge/master
Fixed/improved error parsing from API messages.pull/3340/merge
commit
a49144a0b2
|
@ -11,6 +11,8 @@
|
||||||
"name" : "test_compile_mbed_dev",
|
"name" : "test_compile_mbed_dev",
|
||||||
"lib" : "mbed-dev"
|
"lib" : "mbed-dev"
|
||||||
}
|
}
|
||||||
]
|
],
|
||||||
|
"target_list" : [],
|
||||||
|
"ignore_list" : []
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -31,7 +31,8 @@
|
||||||
# "name" : "test_compile_mbed_dev",
|
# "name" : "test_compile_mbed_dev",
|
||||||
# "lib" : "mbed-dev"
|
# "lib" : "mbed-dev"
|
||||||
# }
|
# }
|
||||||
# ]
|
# ],
|
||||||
|
# "target_list" : []
|
||||||
#}
|
#}
|
||||||
#
|
#
|
||||||
# The mbed_repo_path field should be changed to point to where your local
|
# The mbed_repo_path field should be changed to point to where your local
|
||||||
|
@ -41,6 +42,10 @@
|
||||||
# "test_compile_mbed_lib" and "test_compile_mbed_dev"
|
# "test_compile_mbed_lib" and "test_compile_mbed_dev"
|
||||||
# The lib field in each says which type of mbed 2 library the app contains.
|
# The lib field in each says which type of mbed 2 library the app contains.
|
||||||
# These test apps MUST be available as repos in the user's online Mercurial area.
|
# These test apps MUST be available as repos in the user's online Mercurial area.
|
||||||
|
# The target_list allows the user to override the set of targets/platforms used
|
||||||
|
# for the compilation.
|
||||||
|
# E.g to just compile for 2 targets, K64F and K22F :
|
||||||
|
# "target_list" : ["K64F", "K22F"]
|
||||||
#
|
#
|
||||||
# Run the script from the mbed-os directory as follows:
|
# Run the script from the mbed-os directory as follows:
|
||||||
# > python tools/check_release.py
|
# > python tools/check_release.py
|
||||||
|
@ -51,8 +56,8 @@
|
||||||
# The lib files within the test apps are then updated to the corresponding version in
|
# The lib files within the test apps are then updated to the corresponding version in
|
||||||
# the associated lib itself. The test apps are then committed and pushed back to the users
|
# the associated lib itself. The test apps are then committed and pushed back to the users
|
||||||
# fork.
|
# fork.
|
||||||
# The test apps will then be compiled for all supported targets and a % result output at the
|
# The test apps will then be compiled for all supported targets and a % result output at
|
||||||
# end.
|
# the end.
|
||||||
#
|
#
|
||||||
# Uses the online compiler API at https://mbed.org/handbook/Compile-API
|
# Uses the online compiler API at https://mbed.org/handbook/Compile-API
|
||||||
# Based on the example from https://mbed.org/teams/mbed/code/mbed-API-helper/
|
# Based on the example from https://mbed.org/teams/mbed/code/mbed-API-helper/
|
||||||
|
@ -76,21 +81,39 @@ OFFICIAL_MBED_LIBRARY_BUILD = get_mbed_official_release('2')
|
||||||
|
|
||||||
def get_compilation_failure(messages):
|
def get_compilation_failure(messages):
|
||||||
""" Reads the json formatted 'messages' and checks for compilation errors.
|
""" Reads the json formatted 'messages' and checks for compilation errors.
|
||||||
If there is a genuine compilation error then there should be a new message containing
|
If there is a genuine compilation error then there should be a new
|
||||||
a severity field = Error and an accompanying message with the compile error text.
|
message containing a severity field = Error and an accompanying message
|
||||||
Any other combination is considered an internal compile engine failure
|
with the compile error text. Any other combination is considered an
|
||||||
|
internal compile engine failure
|
||||||
Args:
|
Args:
|
||||||
messages - json formatted text returned by the online compiler API.
|
messages - json formatted text returned by the online compiler API.
|
||||||
|
|
||||||
Returns:
|
Returns:
|
||||||
Either a string containing a compilation error or "Internal" to indicate an error with
|
Either "Error" or "Internal" to indicate an actual compilation error or an
|
||||||
the online IDE API itself.
|
internal IDE API fault.
|
||||||
|
|
||||||
"""
|
"""
|
||||||
for m in messages:
|
for m in messages:
|
||||||
if 'severity' in m and 'message' in m:
|
# Get message text if it exists
|
||||||
if m['severity'] == 'error':
|
try:
|
||||||
return m['message']
|
message = m['message']
|
||||||
|
message = message + "\n"
|
||||||
|
except KeyError:
|
||||||
|
# Skip this message as it has no 'message' field
|
||||||
|
continue
|
||||||
|
|
||||||
|
# Get type of message text
|
||||||
|
try:
|
||||||
|
msg_type = m['type']
|
||||||
|
except KeyError:
|
||||||
|
# Skip this message as it has no 'type' field
|
||||||
|
continue
|
||||||
|
|
||||||
|
if msg_type == 'error' or msg_type == 'tool_error':
|
||||||
|
rel_log.error(message)
|
||||||
|
return "Error"
|
||||||
|
else:
|
||||||
|
rel_log.debug(message)
|
||||||
|
|
||||||
return "Internal"
|
return "Internal"
|
||||||
|
|
||||||
|
@ -111,48 +134,56 @@ def invoke_api(payload, url, auth, polls, begin="start/"):
|
||||||
"""
|
"""
|
||||||
|
|
||||||
# send task to api
|
# send task to api
|
||||||
logging.debug(url + begin + "| data: " + str(payload))
|
rel_log.debug(url + begin + "| data: " + str(payload))
|
||||||
r = requests.post(url + begin, data=payload, auth=auth)
|
r = requests.post(url + begin, data=payload, auth=auth)
|
||||||
logging.debug(r.request.body)
|
rel_log.debug(r.request.body)
|
||||||
|
|
||||||
if r.status_code != 200:
|
if r.status_code != 200:
|
||||||
logging.error("HTTP code %d reported.", r.status_code)
|
rel_log.error("HTTP code %d reported.", r.status_code)
|
||||||
return False, "Internal"
|
return False, "Internal"
|
||||||
|
|
||||||
response = r.json()
|
response = r.json()
|
||||||
logging.debug(response)
|
rel_log.debug(response)
|
||||||
uuid = response['result']['data']['task_id']
|
uuid = response['result']['data']['task_id']
|
||||||
logging.debug("Task accepted and given ID: %s", uuid)
|
rel_log.debug("Task accepted and given ID: %s", uuid)
|
||||||
result = False
|
result = False
|
||||||
fail_type = None
|
fail_type = None
|
||||||
|
|
||||||
# It currently seems to take the onlide IDE API ~30s to process the compile request and
|
# It currently seems to take the onlide IDE API ~30s to process the compile
|
||||||
# provide a response. Set the poll time to half that in case it does manage to compile
|
# request and provide a response. Set the poll time to half that in case it
|
||||||
# quicker.
|
# does manage to compile quicker.
|
||||||
poll_delay = 15
|
poll_delay = 15
|
||||||
logging.debug("Running with a poll for response delay of: %ss", poll_delay)
|
rel_log.debug("Running with a poll for response delay of: %ss", poll_delay)
|
||||||
|
|
||||||
# poll for output
|
# poll for output
|
||||||
for check in range(polls):
|
for check in range(polls):
|
||||||
time.sleep(poll_delay)
|
time.sleep(poll_delay)
|
||||||
r = requests.get(url + "output/%s" % uuid, auth=auth)
|
|
||||||
response = r.json()
|
try:
|
||||||
if response['result']['data']['task_complete']:
|
r = requests.get(url + "output/%s" % uuid, auth=auth)
|
||||||
|
|
||||||
|
except ConnectionError:
|
||||||
|
return "Internal"
|
||||||
|
|
||||||
|
response = r.json()
|
||||||
|
|
||||||
|
data = response['result']['data']
|
||||||
|
if data['task_complete']:
|
||||||
# Task completed. Now determine the result. Should be one of :
|
# Task completed. Now determine the result. Should be one of :
|
||||||
# 1) Successful compilation
|
# 1) Successful compilation
|
||||||
# 2) Failed compilation with an error message
|
# 2) Failed compilation with an error message
|
||||||
# 3) Internal failure of the online compiler
|
# 3) Internal failure of the online compiler
|
||||||
result = bool(response['result']['data']['compilation_success'])
|
result = bool(data['compilation_success'])
|
||||||
if result:
|
if result:
|
||||||
logging.info("\t\tCompilation SUCCESSFUL\n")
|
rel_log.info("COMPILATION SUCCESSFUL\n")
|
||||||
else:
|
else:
|
||||||
# Did this fail due to a genuine compilation error or a failue of the api itself ?
|
# Did this fail due to a genuine compilation error or a failue of
|
||||||
logging.info("\t\tCompilation FAILURE\n")
|
# the api itself ?
|
||||||
fail_type = get_compilation_failure(response['result']['data']['new_messages'])
|
rel_log.info("COMPILATION FAILURE\n")
|
||||||
|
fail_type = get_compilation_failure(data['new_messages'])
|
||||||
break
|
break
|
||||||
else:
|
else:
|
||||||
logging.info("\t\tCompilation FAILURE\n")
|
rel_log.info("COMPILATION FAILURE\n")
|
||||||
|
|
||||||
if not result and fail_type == None:
|
if not result and fail_type == None:
|
||||||
fail_type = "Internal"
|
fail_type = "Internal"
|
||||||
|
@ -160,8 +191,10 @@ def invoke_api(payload, url, auth, polls, begin="start/"):
|
||||||
return result, fail_type
|
return result, fail_type
|
||||||
|
|
||||||
|
|
||||||
def build_repo(target, program, user, pw, polls=25, url="https://developer.mbed.org/api/v2/tasks/compiler/"):
|
def build_repo(target, program, user, pw, polls=25,
|
||||||
""" Wrapper for sending an API command request to the online IDE. Sends a build request.
|
url="https://developer.mbed.org/api/v2/tasks/compiler/"):
|
||||||
|
""" Wrapper for sending an API command request to the online IDE. Sends a
|
||||||
|
build request.
|
||||||
|
|
||||||
Args:
|
Args:
|
||||||
target - Target to be built
|
target - Target to be built
|
||||||
|
@ -192,11 +225,12 @@ def run_cmd(command, exit_on_failure=False):
|
||||||
Returns:
|
Returns:
|
||||||
result - True/False indicating the success/failure of the command
|
result - True/False indicating the success/failure of the command
|
||||||
"""
|
"""
|
||||||
logging.debug('[Exec] %s', ' '.join(command))
|
rel_log.debug('[Exec] %s', ' '.join(command))
|
||||||
return_code = subprocess.call(command, shell=True)
|
return_code = subprocess.call(command, shell=True)
|
||||||
|
|
||||||
if return_code:
|
if return_code:
|
||||||
logging.warning("The command '%s' failed with return code: %s", (' '.join(command), return_code))
|
rel_log.warning("The command '%s' failed with return code: %s",
|
||||||
|
(' '.join(command), return_code))
|
||||||
if exit_on_failure:
|
if exit_on_failure:
|
||||||
sys.exit(1)
|
sys.exit(1)
|
||||||
|
|
||||||
|
@ -217,22 +251,24 @@ def run_cmd_with_output(command, exit_on_failure=False):
|
||||||
result - True/False indicating the success/failure of the command
|
result - True/False indicating the success/failure of the command
|
||||||
output - The output of the command if it was successful, else empty string
|
output - The output of the command if it was successful, else empty string
|
||||||
"""
|
"""
|
||||||
logging.debug('[Exec] %s', ' '.join(command))
|
rel_log.debug('[Exec] %s', ' '.join(command))
|
||||||
returncode = 0
|
returncode = 0
|
||||||
output = ""
|
output = ""
|
||||||
try:
|
try:
|
||||||
output = subprocess.check_output(command, shell=True)
|
output = subprocess.check_output(command, shell=True)
|
||||||
except subprocess.CalledProcessError as e:
|
except subprocess.CalledProcessError as e:
|
||||||
logging.warning("The command '%s' failed with return code: %s", (' '.join(command), e.returncode))
|
rel_log.warning("The command '%s' failed with return code: %s",
|
||||||
|
(' '.join(command), e.returncode))
|
||||||
returncode = e.returncode
|
returncode = e.returncode
|
||||||
if exit_on_failure:
|
if exit_on_failure:
|
||||||
sys.exit(1)
|
sys.exit(1)
|
||||||
return returncode, output
|
return returncode, output
|
||||||
|
|
||||||
def upgrade_test_repo(test, user, library, ref, repo_path):
|
def upgrade_test_repo(test, user, library, ref, repo_path):
|
||||||
""" Upgrades a local version of a test repo to the latest version of its embedded library.
|
""" Upgrades a local version of a test repo to the latest version of its
|
||||||
If the test repo is not present in the user area specified in the json config file, then
|
embedded library.
|
||||||
it will first be cloned.
|
If the test repo is not present in the user area specified in the json
|
||||||
|
config file, then it will first be cloned.
|
||||||
Args:
|
Args:
|
||||||
test - Mercurial test repo name
|
test - Mercurial test repo name
|
||||||
user - Mercurial user name
|
user - Mercurial user name
|
||||||
|
@ -243,7 +279,7 @@ def upgrade_test_repo(test, user, library, ref, repo_path):
|
||||||
Returns:
|
Returns:
|
||||||
updated - True if library was updated, False otherwise
|
updated - True if library was updated, False otherwise
|
||||||
"""
|
"""
|
||||||
logging.info("Updating test repo: '%s' to SHA: %s", test, ref)
|
rel_log.info("Updating test repo: '%s' to SHA: %s", test, ref)
|
||||||
cwd = os.getcwd()
|
cwd = os.getcwd()
|
||||||
|
|
||||||
repo = "https://" + user + '@developer.mbed.org/users/' + user + '/code/' + test
|
repo = "https://" + user + '@developer.mbed.org/users/' + user + '/code/' + test
|
||||||
|
@ -251,7 +287,7 @@ def upgrade_test_repo(test, user, library, ref, repo_path):
|
||||||
# Clone the repo if it doesn't already exist
|
# Clone the repo if it doesn't already exist
|
||||||
path = abspath(repo_path + '/' + test)
|
path = abspath(repo_path + '/' + test)
|
||||||
if not os.path.exists(path):
|
if not os.path.exists(path):
|
||||||
logging.info("Test repo doesn't exist, cloning...")
|
rel_log.info("Test repo doesn't exist, cloning...")
|
||||||
os.chdir(abspath(repo_path))
|
os.chdir(abspath(repo_path))
|
||||||
clone_cmd = ['hg', 'clone', repo]
|
clone_cmd = ['hg', 'clone', repo]
|
||||||
run_cmd(clone_cmd, exit_on_failure=True)
|
run_cmd(clone_cmd, exit_on_failure=True)
|
||||||
|
@ -270,7 +306,7 @@ def upgrade_test_repo(test, user, library, ref, repo_path):
|
||||||
|
|
||||||
os.rename(lib_file, bak_file)
|
os.rename(lib_file, bak_file)
|
||||||
else:
|
else:
|
||||||
logging.error("!! Error trying to backup lib file prior to updating.")
|
rel_log.error("Failure to backup lib file prior to updating.")
|
||||||
return False
|
return False
|
||||||
|
|
||||||
# mbed 2 style lib file contains one line with the following format
|
# mbed 2 style lib file contains one line with the following format
|
||||||
|
@ -279,7 +315,8 @@ def upgrade_test_repo(test, user, library, ref, repo_path):
|
||||||
lib_re = re.compile(exp)
|
lib_re = re.compile(exp)
|
||||||
updated = False
|
updated = False
|
||||||
|
|
||||||
# Scan through mbed-os.lib line by line, looking for lib version and update it if found
|
# Scan through mbed-os.lib line by line, looking for lib version and update
|
||||||
|
# it if found
|
||||||
with open(bak_file, 'r') as ip, open(lib_file, 'w') as op:
|
with open(bak_file, 'r') as ip, open(lib_file, 'w') as op:
|
||||||
for line in ip:
|
for line in ip:
|
||||||
|
|
||||||
|
@ -297,8 +334,9 @@ def upgrade_test_repo(test, user, library, ref, repo_path):
|
||||||
# Setup the default commit message
|
# Setup the default commit message
|
||||||
commit_message = '"Updating ' + library + ' to ' + ref + '"'
|
commit_message = '"Updating ' + library + ' to ' + ref + '"'
|
||||||
|
|
||||||
# Setup and run the commit command. Need to use the rawcommand in the hglib for this in order to pass
|
# Setup and run the commit command. Need to use the rawcommand in the hglib
|
||||||
# the string value to the -m option. run_cmd using subprocess does not like this syntax.
|
# for this in order to pass the string value to the -m option. run_cmd using
|
||||||
|
# subprocess does not like this syntax.
|
||||||
try:
|
try:
|
||||||
client.rawcommand(['commit','-m '+commit_message, lib_file])
|
client.rawcommand(['commit','-m '+commit_message, lib_file])
|
||||||
|
|
||||||
|
@ -306,7 +344,7 @@ def upgrade_test_repo(test, user, library, ref, repo_path):
|
||||||
run_cmd(cmd, exit_on_failure=True)
|
run_cmd(cmd, exit_on_failure=True)
|
||||||
|
|
||||||
except:
|
except:
|
||||||
logging.info("Lib file already up to date and thus nothing to commit")
|
rel_log.info("Lib file already up to date and thus nothing to commit")
|
||||||
|
|
||||||
os.chdir(cwd)
|
os.chdir(cwd)
|
||||||
return updated
|
return updated
|
||||||
|
@ -362,11 +400,21 @@ def get_latest_library_versions(repo_path):
|
||||||
|
|
||||||
return mbed, mbed_dev
|
return mbed, mbed_dev
|
||||||
|
|
||||||
|
def log_results(lst, title):
|
||||||
|
if len(lst) == 0:
|
||||||
|
rel_log.info("%s - None", title)
|
||||||
|
else:
|
||||||
|
for entry in lst:
|
||||||
|
rel_log.info("%s - Test: %s, Target: %s", title, entry[0], entry[1])
|
||||||
|
|
||||||
|
|
||||||
if __name__ == '__main__':
|
if __name__ == '__main__':
|
||||||
|
|
||||||
parser = argparse.ArgumentParser(description=__doc__,
|
parser = argparse.ArgumentParser(description=__doc__,
|
||||||
formatter_class=argparse.RawDescriptionHelpFormatter)
|
formatter_class=argparse.RawDescriptionHelpFormatter)
|
||||||
parser.add_argument('-l', '--log-level', help="Level for providing logging output", default='INFO')
|
parser.add_argument('-l', '--log-level',
|
||||||
|
help="Level for providing logging output",
|
||||||
|
default='INFO')
|
||||||
args = parser.parse_args()
|
args = parser.parse_args()
|
||||||
|
|
||||||
default = getattr(logging, 'INFO')
|
default = getattr(logging, 'INFO')
|
||||||
|
@ -374,17 +422,28 @@ if __name__ == '__main__':
|
||||||
|
|
||||||
# Set logging level
|
# Set logging level
|
||||||
logging.basicConfig(level=level)
|
logging.basicConfig(level=level)
|
||||||
|
rel_log = logging.getLogger("check-release")
|
||||||
|
|
||||||
# Read configuration data
|
# Read configuration data
|
||||||
json_data = json.load(open(os.path.join(os.path.dirname(__file__), "check_release.json")))
|
with open(os.path.join(os.path.dirname(__file__), "check_release.json")) as config:
|
||||||
|
json_data = json.load(config)
|
||||||
|
|
||||||
supported_targets = []
|
supported_targets = []
|
||||||
|
|
||||||
# Get a list of the officially supported mbed-os 2 targets
|
if len(json_data["target_list"]) > 0:
|
||||||
for tgt in OFFICIAL_MBED_LIBRARY_BUILD:
|
# Compile user supplied subset of targets
|
||||||
supported_targets.append(tgt[0])
|
supported_targets = json_data["target_list"]
|
||||||
|
else:
|
||||||
|
# Get a list of the officially supported mbed-os 2 targets
|
||||||
|
for tgt in OFFICIAL_MBED_LIBRARY_BUILD:
|
||||||
|
supported_targets.append(tgt[0])
|
||||||
|
|
||||||
|
ignore_list = []
|
||||||
|
|
||||||
|
if len(json_data["ignore_list"]) > 0:
|
||||||
|
# List of tuples of (test, target) to be ignored in this test
|
||||||
|
ignore_list = json_data["ignore_list"]
|
||||||
|
|
||||||
config = json_data["config"]
|
config = json_data["config"]
|
||||||
test_list = json_data["test_list"]
|
test_list = json_data["test_list"]
|
||||||
repo_path = config["mbed_repo_path"]
|
repo_path = config["mbed_repo_path"]
|
||||||
|
@ -405,44 +464,67 @@ if __name__ == '__main__':
|
||||||
mbed, mbed_dev = get_latest_library_versions(repo_path)
|
mbed, mbed_dev = get_latest_library_versions(repo_path)
|
||||||
|
|
||||||
if not mbed or not mbed_dev:
|
if not mbed or not mbed_dev:
|
||||||
logging.error("Could not obtain latest versions of library files!!")
|
rel_log.error("Could not obtain latest versions of library files!!")
|
||||||
exit(1)
|
exit(1)
|
||||||
|
|
||||||
logging.info("Latest mbed lib version = %s", mbed)
|
rel_log.info("Latest mbed lib version = %s", mbed)
|
||||||
logging.info("Latest mbed-dev lib version = %s", mbed_dev)
|
rel_log.info("Latest mbed-dev lib version = %s", mbed_dev)
|
||||||
|
|
||||||
# First update test repos to latest versions of their embedded libraries
|
# First update test repos to latest versions of their embedded libraries
|
||||||
for test in test_list:
|
for test in test_list:
|
||||||
tests.append(test['name'])
|
tests.append(test['name'])
|
||||||
upgrade_test_repo(test['name'], user, test['lib'], mbed if test['lib'] == "mbed" else mbed_dev, repo_path)
|
upgrade_test_repo(test['name'], user, test['lib'],
|
||||||
|
mbed if test['lib'] == "mbed" else mbed_dev,
|
||||||
total = len(supported_targets)*len(tests)
|
repo_path)
|
||||||
|
|
||||||
|
total = len(supported_targets) * len(tests)
|
||||||
|
current = 0
|
||||||
retries = 10
|
retries = 10
|
||||||
passes = 0
|
passes = 0
|
||||||
|
failures = []
|
||||||
|
skipped = []
|
||||||
|
|
||||||
# Compile each test for each supported target
|
# Compile each test for each supported target
|
||||||
for test in tests:
|
for test in tests:
|
||||||
logging.info("Test compiling program: %s\n", test)
|
|
||||||
for target in supported_targets:
|
for target in supported_targets:
|
||||||
|
|
||||||
|
combo = [test, target]
|
||||||
|
|
||||||
|
if combo in ignore_list:
|
||||||
|
rel_log.info("SKIPPING TEST: %s, TARGET: %s", test, target)
|
||||||
|
total -= 1
|
||||||
|
skipped.append(combo)
|
||||||
|
continue
|
||||||
|
|
||||||
|
current += 1
|
||||||
for retry in range(0, retries):
|
for retry in range(0, retries):
|
||||||
logging.info("\tCompiling target: %s , attempt %u\n", target, retry)
|
rel_log.info("COMPILING (%d/%d): TEST %s, TARGET: %s , attempt %u\n", current, total, test, target, retry)
|
||||||
result, mesg = build_repo(target, test, user, password)
|
result, mesg = build_repo(target, test, user, password)
|
||||||
if not result:
|
if not result:
|
||||||
if mesg == 'Internal':
|
if mesg == 'Internal':
|
||||||
# Internal compiler error thus retry
|
# Internal compiler error thus retry
|
||||||
continue
|
continue
|
||||||
else:
|
else:
|
||||||
# Genuine compilation error, thus print it out
|
# Actual error thus move on to next compilation
|
||||||
logging.error("\t\tError: %s\n", mesg)
|
failures.append(combo)
|
||||||
|
break
|
||||||
|
|
||||||
passes += (int)(result)
|
passes += (int)(result)
|
||||||
break
|
break
|
||||||
else:
|
else:
|
||||||
logging.error("\t\tProgram/Target compilation failed due to internal errors. Removing from considered list!\n")
|
rel_log.error("Compilation failed due to internal errors.")
|
||||||
|
rel_log.error("Skipping test/target combination.")
|
||||||
total -= 1
|
total -= 1
|
||||||
|
skipped.append(combo)
|
||||||
|
|
||||||
|
rel_log.info(" SUMMARY OF COMPILATION RESULTS")
|
||||||
|
rel_log.info(" ------------------------------")
|
||||||
|
rel_log.info(" NUMBER OF TEST APPS: %d, NUMBER OF TARGETS: %d",
|
||||||
|
len(tests), len(supported_targets))
|
||||||
|
log_results(failures, " FAILED")
|
||||||
|
log_results(skipped, " SKIPPED")
|
||||||
|
|
||||||
# Output a % pass rate, indicate a failure if not 100% successful
|
# Output a % pass rate, indicate a failure if not 100% successful
|
||||||
pass_rate = int(passes/total) * 100
|
pass_rate = (float(passes) / float(total)) * 100.0
|
||||||
logging.info("Pass percentage = %d\n", pass_rate)
|
rel_log.info(" PASS RATE %.1f %%\n", pass_rate)
|
||||||
sys.exit(not (pass_rate == 100))
|
sys.exit(not (pass_rate == 100))
|
||||||
|
|
Loading…
Reference in New Issue