blob: 02d9c49323da594eb39b21b80148065d518cf56d [file] [log] [blame]
# tests/__init__.py -- Portage Unit Test functionality
# Copyright 2006-2021 Gentoo Authors
# Distributed under the terms of the GNU General Public License v2
import argparse
import sys
import time
import unittest
from pathlib import Path
from unittest.runner import TextTestResult as _TextTestResult
import portage
from portage import os
from portage import _encodings
from portage import _unicode_decode
from portage.proxy.objectproxy import ObjectProxy
# This remains constant when the real value is a mock.
EPREFIX_ORIG = portage.const.EPREFIX
class lazy_value(ObjectProxy):
__slots__ = ("_func",)
def __init__(self, func):
ObjectProxy.__init__(self)
object.__setattr__(self, "_func", func)
def _get_target(self):
return object.__getattribute__(self, "_func")()
@lazy_value
def cnf_path():
if portage._not_installed:
return os.path.join(portage.const.PORTAGE_BASE_PATH, "cnf")
return os.path.join(
EPREFIX_ORIG or "/", portage.const.GLOBAL_CONFIG_PATH.lstrip(os.sep)
)
@lazy_value
def cnf_etc_path():
if portage._not_installed:
return str(cnf_path)
return os.path.join(EPREFIX_ORIG or "/", "etc")
@lazy_value
def cnf_bindir():
if portage._not_installed:
return portage.const.PORTAGE_BIN_PATH
return os.path.join(portage.const.EPREFIX or "/", "usr", "bin")
@lazy_value
def cnf_sbindir():
if portage._not_installed:
return str(cnf_bindir)
return os.path.join(portage.const.EPREFIX or "/", "usr", "sbin")
def main():
suite = unittest.TestSuite()
basedir = Path(__file__).resolve().parent
argv0 = Path(sys.argv[0])
usage = "usage: %s [options] [tests to run]" % argv0.name
parser = argparse.ArgumentParser(usage=usage)
parser.add_argument(
"-l", "--list", help="list all tests", action="store_true", dest="list_tests"
)
parser.add_argument("tests", nargs="*", type=Path)
options = parser.parse_args(args=sys.argv)
if (
os.environ.get("NOCOLOR") in ("yes", "true")
or os.environ.get("TERM") == "dumb"
or not sys.stdout.isatty()
):
portage.output.nocolor()
if options.list_tests:
testdir = argv0.parent
for mydir in getTestDirs(basedir):
testsubdir = mydir.name
for name in getTestNames(mydir):
print("%s/%s/%s.py" % (testdir, testsubdir, name))
return os.EX_OK
if len(options.tests) > 1:
suite.addTests(getTestFromCommandLine(options.tests[1:], basedir))
else:
for mydir in getTestDirs(basedir):
suite.addTests(getTests(mydir, basedir))
result = TextTestRunner(verbosity=2).run(suite)
if not result.wasSuccessful():
return 1
return os.EX_OK
def my_import(name):
mod = __import__(name)
components = name.split(".")
for comp in components[1:]:
mod = getattr(mod, comp)
return mod
def getTestFromCommandLine(args, base_path):
result = []
for arg in args:
realpath = arg.resolve()
path = realpath.parent
f = realpath.relative_to(path)
if not f.name.startswith("test") or not f.suffix == ".py":
raise Exception("Invalid argument: '%s'" % arg)
mymodule = f.stem
result.extend(getTestsFromFiles(path, base_path, [mymodule]))
return result
def getTestDirs(base_path):
TEST_FILE = "__test__.py"
testDirs = []
# the os.walk help mentions relative paths as being quirky
# I was tired of adding dirs to the list, so now we add __test__.py
# to each dir we want tested.
for testFile in base_path.rglob(TEST_FILE):
testDirs.append(testFile.parent)
testDirs.sort()
return testDirs
def getTestNames(path):
files = path.glob("*")
files = [f.stem for f in files if f.name.startswith("test") and f.suffix == ".py"]
files.sort()
return files
def getTestsFromFiles(path, base_path, files):
parent_path = path.relative_to(base_path)
parent_module = ".".join(("portage", "tests") + parent_path.parts)
result = []
for mymodule in files:
# Make the trailing / a . for module importing
modname = ".".join((parent_module, mymodule))
mod = my_import(modname)
result.append(unittest.TestLoader().loadTestsFromModule(mod))
return result
def getTests(path, base_path):
"""
path is the path to a given subdir ( 'portage/' for example)
This does a simple filter on files in that dir to give us modules
to import
"""
return getTestsFromFiles(path, base_path, getTestNames(path))
class TextTestResult(_TextTestResult):
"""
We need a subclass of unittest.runner.TextTestResult to handle tests with TODO
This just adds an addTodo method that can be used to add tests
that are marked TODO; these can be displayed later
by the test runner.
"""
def __init__(self, stream, descriptions, verbosity):
super(TextTestResult, self).__init__(stream, descriptions, verbosity)
self.todoed = []
self.portage_skipped = []
def addTodo(self, test, info):
self.todoed.append((test, info))
if self.showAll:
self.stream.writeln("TODO")
elif self.dots:
self.stream.write(".")
def addPortageSkip(self, test, info):
self.portage_skipped.append((test, info))
if self.showAll:
self.stream.writeln("SKIP")
elif self.dots:
self.stream.write(".")
def printErrors(self):
if self.dots or self.showAll:
self.stream.writeln()
self.printErrorList("ERROR", self.errors)
self.printErrorList("FAIL", self.failures)
self.printErrorList("TODO", self.todoed)
self.printErrorList("SKIP", self.portage_skipped)
class TestCase(unittest.TestCase):
"""
We need a way to mark a unit test as "ok to fail"
This way someone can add a broken test and mark it as failed
and then fix the code later. This may not be a great approach
(broken code!!??!11oneone) but it does happen at times.
"""
def __init__(self, *pargs, **kwargs):
unittest.TestCase.__init__(self, *pargs, **kwargs)
self.todo = False
self.portage_skip = None
self.cnf_path = cnf_path
self.cnf_etc_path = cnf_etc_path
self.bindir = cnf_bindir
self.sbindir = cnf_sbindir
def defaultTestResult(self):
return TextTestResult()
def run(self, result=None):
if result is None:
result = self.defaultTestResult()
result.startTest(self)
testMethod = getattr(self, self._testMethodName)
try:
ok = False
try:
try:
self.setUp()
except KeyboardInterrupt:
raise
except unittest.SkipTest:
raise
except Exception:
result.addError(self, sys.exc_info())
return
testMethod()
ok = True
except unittest.SkipTest as e:
result.addPortageSkip(self, "%s: SKIP: %s" % (testMethod, str(e)))
except self.failureException:
if self.portage_skip is not None:
if self.portage_skip is True:
result.addPortageSkip(self, "%s: SKIP" % testMethod)
else:
result.addPortageSkip(
self, "%s: SKIP: %s" % (testMethod, self.portage_skip)
)
elif self.todo:
result.addTodo(self, "%s: TODO" % testMethod)
else:
result.addFailure(self, sys.exc_info())
except (KeyboardInterrupt, SystemExit):
raise
except:
result.addError(self, sys.exc_info())
try:
self.tearDown()
except SystemExit:
raise
except KeyboardInterrupt:
raise
except:
result.addError(self, sys.exc_info())
ok = False
if ok:
result.addSuccess(self)
finally:
result.stopTest(self)
def assertRaisesMsg(self, msg, excClass, callableObj, *args, **kwargs):
"""Fail unless an exception of class excClass is thrown
by callableObj when invoked with arguments args and keyword
arguments kwargs. If a different type of exception is
thrown, it will not be caught, and the test case will be
deemed to have suffered an error, exactly as for an
unexpected exception.
"""
try:
callableObj(*args, **kwargs)
except excClass:
return
else:
if hasattr(excClass, "__name__"):
excName = excClass.__name__
else:
excName = str(excClass)
raise self.failureException("%s not raised: %s" % (excName, msg))
def assertNotExists(self, path):
"""Make sure |path| does not exist"""
path = Path(path)
if path.exists():
raise self.failureException("path exists when it should not: %s" % path)
class TextTestRunner(unittest.TextTestRunner):
"""
We subclass unittest.TextTestRunner to output SKIP for tests that fail but are skippable
"""
def _makeResult(self):
return TextTestResult(self.stream, self.descriptions, self.verbosity)
def run(self, test):
"""
Run the given test case or test suite.
"""
result = self._makeResult()
startTime = time.time()
test(result)
stopTime = time.time()
timeTaken = stopTime - startTime
result.printErrors()
self.stream.writeln(result.separator2)
run = result.testsRun
self.stream.writeln(
"Ran %d test%s in %.3fs" % (run, run != 1 and "s" or "", timeTaken)
)
self.stream.writeln()
if not result.wasSuccessful():
self.stream.write("FAILED (")
failed = len(result.failures)
errored = len(result.errors)
if failed:
self.stream.write("failures=%d" % failed)
if errored:
if failed:
self.stream.write(", ")
self.stream.write("errors=%d" % errored)
self.stream.writeln(")")
else:
self.stream.writeln("OK")
return result
test_cps = ["sys-apps/portage", "virtual/portage"]
test_versions = ["1.0", "1.0-r1", "2.3_p4", "1.0_alpha57"]
test_slots = [None, "1", "gentoo-sources-2.6.17", "spankywashere"]
test_usedeps = ["foo", "-bar", ("foo", "bar"), ("foo", "-bar"), ("foo?", "!bar?")]