mirror of
git://git.yoctoproject.org/poky.git
synced 2025-07-19 21:09:03 +02:00

Since patchtest is in oe-core, the Python os module's methods can be used to retrieve the repository path and tests directory by default. This reduces the number of mandatory arguments for invocation of patchtest unless the user wants to use a custom test suite or test patches against a different repo. The REPO and TESTDIR arguments are likewise adjusted so that they are optional. Also, make it more obvious what the --startdir flag is meant for on the command line by renaming it to --testdir, and update the scripts/patchtest.README file to be consistent with the new usage. (From OE-Core rev: bae7421ece4806f5148f164293810b9fe75e0756) Signed-off-by: Trevor Gamblin <tgamblin@baylibre.com> Signed-off-by: Richard Purdie <richard.purdie@linuxfoundation.org>
234 lines
7.5 KiB
Python
Executable File
234 lines
7.5 KiB
Python
Executable File
#!/usr/bin/env python3
|
|
# ex:ts=4:sw=4:sts=4:et
|
|
# -*- tab-width: 4; c-basic-offset: 4; indent-tabs-mode: nil -*-
|
|
#
|
|
# patchtest: execute all unittest test cases discovered for a single patch
|
|
#
|
|
# Copyright (C) 2016 Intel Corporation
|
|
#
|
|
# This program is free software; you can redistribute it and/or modify
|
|
# it under the terms of the GNU General Public License version 2 as
|
|
# published by the Free Software Foundation.
|
|
#
|
|
# This program is distributed in the hope that it will be useful,
|
|
# but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
# GNU General Public License for more details.
|
|
#
|
|
# You should have received a copy of the GNU General Public License along
|
|
# with this program; if not, write to the Free Software Foundation, Inc.,
|
|
# 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA.
|
|
#
|
|
# Author: Leo Sandoval <leonardo.sandoval.gonzalez@linux.intel.com>
|
|
#
|
|
|
|
import sys
|
|
import os
|
|
import unittest
|
|
import fileinput
|
|
import logging
|
|
import traceback
|
|
import json
|
|
|
|
# Include current path so test cases can see it
|
|
sys.path.insert(0, os.path.dirname(os.path.realpath(__file__)))
|
|
|
|
# Include patchtest library
|
|
sys.path.insert(0, os.path.join(os.path.dirname(os.path.realpath(__file__)), '../meta/lib/patchtest'))
|
|
|
|
from data import PatchTestInput
|
|
from repo import PatchTestRepo
|
|
|
|
import utils
|
|
logger = utils.logger_create('patchtest')
|
|
info = logger.info
|
|
error = logger.error
|
|
|
|
import repo
|
|
|
|
def getResult(patch, mergepatch, logfile=None):
|
|
|
|
class PatchTestResult(unittest.TextTestResult):
|
|
""" Patchtest TextTestResult """
|
|
shouldStop = True
|
|
longMessage = False
|
|
|
|
success = 'PASS'
|
|
fail = 'FAIL'
|
|
skip = 'SKIP'
|
|
|
|
def startTestRun(self):
|
|
# let's create the repo already, it can be used later on
|
|
repoargs = {
|
|
'repodir': PatchTestInput.repodir,
|
|
'commit' : PatchTestInput.basecommit,
|
|
'branch' : PatchTestInput.basebranch,
|
|
'patch' : patch,
|
|
}
|
|
|
|
self.repo_error = False
|
|
self.test_error = False
|
|
self.test_failure = False
|
|
|
|
try:
|
|
self.repo = PatchTestInput.repo = PatchTestRepo(**repoargs)
|
|
except:
|
|
logger.error(traceback.print_exc())
|
|
self.repo_error = True
|
|
self.stop()
|
|
return
|
|
|
|
if mergepatch:
|
|
self.repo.merge()
|
|
|
|
def addError(self, test, err):
|
|
self.test_error = True
|
|
(ty, va, trace) = err
|
|
logger.error(traceback.print_exc())
|
|
|
|
def addFailure(self, test, err):
|
|
test_description = test.id().split('.')[-1].replace('_', ' ').replace("cve", "CVE").replace("signed off by",
|
|
"Signed-off-by").replace("upstream status",
|
|
"Upstream-Status").replace("non auh",
|
|
"non-AUH").replace("presence format", "presence")
|
|
self.test_failure = True
|
|
fail_str = '{}: {}: {} ({})'.format(self.fail,
|
|
test_description, json.loads(str(err[1]))["issue"],
|
|
test.id())
|
|
print(fail_str)
|
|
if logfile:
|
|
with open(logfile, "a") as f:
|
|
f.write(fail_str + "\n")
|
|
|
|
def addSuccess(self, test):
|
|
test_description = test.id().split('.')[-1].replace('_', ' ').replace("cve", "CVE").replace("signed off by",
|
|
"Signed-off-by").replace("upstream status",
|
|
"Upstream-Status").replace("non auh",
|
|
"non-AUH").replace("presence format", "presence")
|
|
success_str = '{}: {} ({})'.format(self.success,
|
|
test_description, test.id())
|
|
print(success_str)
|
|
if logfile:
|
|
with open(logfile, "a") as f:
|
|
f.write(success_str + "\n")
|
|
|
|
def addSkip(self, test, reason):
|
|
test_description = test.id().split('.')[-1].replace('_', ' ').replace("cve", "CVE").replace("signed off by",
|
|
"Signed-off-by").replace("upstream status",
|
|
"Upstream-Status").replace("non auh",
|
|
"non-AUH").replace("presence format", "presence")
|
|
skip_str = '{}: {}: {} ({})'.format(self.skip,
|
|
test_description, json.loads(str(reason))["issue"],
|
|
test.id())
|
|
print(skip_str)
|
|
if logfile:
|
|
with open(logfile, "a") as f:
|
|
f.write(skip_str + "\n")
|
|
|
|
def stopTestRun(self):
|
|
|
|
# in case there was an error on repo object creation, just return
|
|
if self.repo_error:
|
|
return
|
|
|
|
self.repo.clean()
|
|
|
|
return PatchTestResult
|
|
|
|
def _runner(resultklass, prefix=None):
|
|
# load test with the corresponding prefix
|
|
loader = unittest.TestLoader()
|
|
if prefix:
|
|
loader.testMethodPrefix = prefix
|
|
|
|
# create the suite with discovered tests and the corresponding runner
|
|
suite = loader.discover(start_dir=PatchTestInput.testdir, pattern=PatchTestInput.pattern, top_level_dir=PatchTestInput.topdir)
|
|
ntc = suite.countTestCases()
|
|
|
|
# if there are no test cases, just quit
|
|
if not ntc:
|
|
return 2
|
|
runner = unittest.TextTestRunner(resultclass=resultklass, verbosity=0)
|
|
|
|
try:
|
|
result = runner.run(suite)
|
|
except:
|
|
logger.error(traceback.print_exc())
|
|
logger.error('patchtest: something went wrong')
|
|
return 1
|
|
|
|
return 0
|
|
|
|
def run(patch, logfile=None):
|
|
""" Load, setup and run pre and post-merge tests """
|
|
# Get the result class and install the control-c handler
|
|
unittest.installHandler()
|
|
|
|
# run pre-merge tests, meaning those methods with 'pretest' as prefix
|
|
premerge_resultklass = getResult(patch, False, logfile)
|
|
premerge_result = _runner(premerge_resultklass, 'pretest')
|
|
|
|
# run post-merge tests, meaning those methods with 'test' as prefix
|
|
postmerge_resultklass = getResult(patch, True, logfile)
|
|
postmerge_result = _runner(postmerge_resultklass, 'test')
|
|
|
|
if premerge_result == 2 and postmerge_result == 2:
|
|
logger.error('patchtest: any test cases found - did you specify the correct suite directory?')
|
|
|
|
return premerge_result or postmerge_result
|
|
|
|
def main():
|
|
tmp_patch = False
|
|
patch_path = PatchTestInput.patch_path
|
|
log_results = PatchTestInput.log_results
|
|
log_path = None
|
|
patch_list = None
|
|
|
|
if os.path.isdir(patch_path):
|
|
patch_list = [os.path.join(patch_path, filename) for filename in os.listdir(patch_path)]
|
|
else:
|
|
patch_list = [patch_path]
|
|
|
|
for patch in patch_list:
|
|
if os.path.getsize(patch) == 0:
|
|
logger.error('patchtest: patch is empty')
|
|
return 1
|
|
|
|
logger.info('Testing patch %s' % patch)
|
|
|
|
if log_results:
|
|
log_path = patch + ".testresult"
|
|
with open(log_path, "a") as f:
|
|
f.write("Patchtest results for patch '%s':\n\n" % patch)
|
|
|
|
try:
|
|
if log_path:
|
|
run(patch, log_path)
|
|
else:
|
|
run(patch)
|
|
finally:
|
|
if tmp_patch:
|
|
os.remove(patch)
|
|
|
|
if __name__ == '__main__':
|
|
ret = 1
|
|
|
|
# Parse the command line arguments and store it on the PatchTestInput namespace
|
|
PatchTestInput.set_namespace()
|
|
|
|
# set debugging level
|
|
if PatchTestInput.debug:
|
|
logger.setLevel(logging.DEBUG)
|
|
|
|
# if topdir not define, default it to testdir
|
|
if not PatchTestInput.topdir:
|
|
PatchTestInput.topdir = PatchTestInput.testdir
|
|
|
|
try:
|
|
ret = main()
|
|
except Exception:
|
|
import traceback
|
|
traceback.print_exc(5)
|
|
|
|
sys.exit(ret)
|