blob: dd3b9c1f9ea0cfd250b5999f70717811dc224baa [file] [log] [blame]
import argparse
import os
import psutil
import textwrap
import time
def positive_int_or_default(default):
def positive_integer(v):
if v is None or v == "":
return default
if int(v) <= 0:
raise ValueError("value must be positive")
return int(v)
return positive_integer
def positive_float_or_default(default):
def positive_float(v):
if v is None or v == "":
return default
if float(v) <= 0:
raise ValueError("value must be positive")
return float(v)
return positive_float
def positive_int_or_auto(v):
if v is None or v in ("", "auto"):
return "auto"
if int(v) <= 0:
raise ValueError("value must be positive or auto")
return int(v)
def int_or_auto(v):
if v is None or v in ("", "auto"):
return "auto"
if int(v) < 0:
raise ValueError("value must be positive or auto")
return int(v)
def int_choice_or_default(options, default):
assert default in options
def choice(v):
if v is None or v == "":
return default
if int(v) in options:
return int(v)
raise ValueError("invalid choice")
return choice
def worker_config(v):
if v is None or v == "":
return 0
if v.startswith("workers "):
return(int(v.split(" ")[1]))
return int(v)
def directory(v):
if not os.path.isdir(v):
raise ValueError(f"provided path '{v}' doesn't exist "
"or is not a directory")
return v
def directory_verify_or_create(v):
if not os.path.isdir(v):
os.mkdir(v)
return v
parser = argparse.ArgumentParser(description="VPP unit tests",
formatter_class=argparse.RawTextHelpFormatter)
parser.add_argument("--failfast", action="store_true",
help="stop running tests on first failure")
parser.add_argument("--test-src-dir", action="append", type=directory,
help="directory containing test files "
"(may be specified multiple times) "
"(VPP_WS_DIR/test is added automatically to the set)")
default_verbose = 0
parser.add_argument("--verbose", action="store", default=default_verbose,
type=int_choice_or_default((0, 1, 2), default_verbose),
help="verbosity setting - 0 - least verbose, "
"2 - most verbose (default: 0)")
default_test_run_timeout = 600
parser.add_argument("--timeout", action="store",
type=positive_int_or_default(default_test_run_timeout),
default=default_test_run_timeout,
metavar="TEST_RUN_TIMEOUT",
help="test run timeout in seconds - per test "
f"(default: {default_test_run_timeout})")
parser.add_argument("--failed-dir", action="store", type=directory,
help="directory containing failed tests")
filter_help_string = """\
expression consists of 3 string selectors separated by '.' separators:
<file>.<class>.<function>
- selectors restrict which files/classes/functions are run
- selector can be replaced with '*' or omitted entirely if not needed
- <file> selector is automatically prepended with 'test_' if required
- '.' separators are required only if selector(s) follow(s)
examples:
1. all of the following expressions are equivalent and will select
all test classes and functions from test_bfd.py:
'test_bfd' 'bfd' 'test_bfd..' 'bfd.' 'bfd.*.*' 'test_bfd.*.*'
2. 'bfd.BFDAPITestCase' selects all tests from test_bfd.py,
which are part of BFDAPITestCase class
3. 'bfd.BFDAPITestCase.test_add_bfd' selects a single test named
test_add_bfd from test_bfd.py/BFDAPITestCase
4. '.*.test_add_bfd' selects all test functions named test_add_bfd
from all files/classes
"""
parser.add_argument("--filter", action="store",
metavar="FILTER_EXPRESSION", help=filter_help_string)
default_retries = 0
parser.add_argument("--retries", action="store", default=default_retries,
type=positive_int_or_default(default_retries),
help="retry failed tests RETRIES times")
parser.add_argument("--step", action="store_true", default=False,
help="enable stepping through tests")
debug_help_string = """\
attach - attach to already running vpp
core - detect coredump and load core in gdb on crash
gdb - print VPP PID and pause allowing attaching gdb
gdbserver - same as above, but run gdb in gdbserver
"""
parser.add_argument("--debug", action="store",
choices=["attach", "core", "gdb", "gdbserver"],
help=debug_help_string)
parser.add_argument("--debug-framework", action="store_true",
help="enable internal test framework debugging")
parser.add_argument("--compress-core", action="store_true",
help="compress core files if not debugging them")
parser.add_argument("--extended", action="store_true",
help="run extended tests")
parser.add_argument("--sanity", action="store_true",
help="perform sanity vpp run before running tests")
parser.add_argument("--force-foreground", action="store_true",
help="force running in foreground - don't fork")
parser.add_argument("--jobs", action="store", type=positive_int_or_auto,
default="auto", help="maximum concurrent test jobs")
parser.add_argument("--venv-dir", action="store",
type=directory, help="path to virtual environment")
default_rnd_seed = time.time()
parser.add_argument("--rnd-seed", action="store", default=default_rnd_seed,
type=positive_float_or_default(default_rnd_seed),
help="random generator seed (default: current time)")
parser.add_argument("--vpp-worker-count", action="store", type=worker_config,
default=0, help="number of vpp workers")
parser.add_argument("--gcov", action="store_true",
default=False, help="running gcov tests")
parser.add_argument("--cache-vpp-output", action="store_true", default=False,
help="cache VPP stdout/stderr and log as one block "
"after test finishes")
parser.add_argument("--vpp-ws-dir", action="store", required=True,
type=directory, help="vpp workspace directory")
parser.add_argument("--vpp-tag", action="store", default="vpp_debug",
metavar="VPP_TAG", required=True,
help="vpp tag (e.g. vpp, vpp_debug, vpp_gcov)")
parser.add_argument("--vpp", action="store", help="path to vpp binary "
"(default: derive from VPP_WS_DIR and VPP_TAG)")
parser.add_argument("--vpp-install-dir", type=directory,
action="store", help="path to vpp install directory"
"(default: derive from VPP_WS_DIR and VPP_TAG)")
parser.add_argument("--vpp-build-dir", action="store", type=directory,
help="vpp build directory"
"(default: derive from VPP_WS_DIR and VPP_TAG)")
parser.add_argument("--vpp-plugin-dir", action="append", type=directory,
help="directory containing vpp plugins"
"(default: derive from VPP_WS_DIR and VPP_TAG)")
parser.add_argument("--vpp-test-plugin-dir", action="append", type=directory,
help="directory containing vpp api test plugins"
"(default: derive from VPP_WS_DIR and VPP_TAG)")
parser.add_argument("--extern-plugin-dir", action="append", type=directory,
default=[], help="directory containing external plugins")
parser.add_argument("--coredump-size", action="store", default="unlimited",
help="specify vpp coredump size")
parser.add_argument("--max-vpp-cpus", action="store", type=int_or_auto,
default=0, help="max cpus used by vpp")
variant_help_string = """\
specify which march node variant to unit test
e.g. --variant=skx - test the skx march variants
e.g. --variant=icl - test the icl march variants
"""
parser.add_argument("--variant", action="store", help=variant_help_string)
parser.add_argument("--api-fuzz", action="store", default=None,
help="specify api fuzzing parameters")
parser.add_argument("--wipe-tmp-dir", action="store_true", default=True,
help="remove test tmp directory before running test")
parser.add_argument("--tmp-dir", action="store", default="/tmp",
type=directory_verify_or_create,
help="directory where to store test temporary directories")
parser.add_argument("--log-dir", action="store",
type=directory_verify_or_create,
help="directory where to store directories "
"containing log files (default: --tmp-dir)")
default_keep_pcaps = False
parser.add_argument("--keep-pcaps", action="store_true",
default=default_keep_pcaps,
help="if set, keep all pcap files from a test run"
f" (default: {default_keep_pcaps})")
config = parser.parse_args()
ws = config.vpp_ws_dir
br = f"{ws}/build-root"
tag = config.vpp_tag
if config.vpp_install_dir is None:
config.vpp_install_dir = f"{br}/install-{tag}-native"
if config.vpp is None:
config.vpp = f"{config.vpp_install_dir}/vpp/bin/vpp"
if config.vpp_build_dir is None:
config.vpp_build_dir = f"{br}/build-{tag}-native"
libs = ["lib", "lib64"]
if config.vpp_plugin_dir is None:
config.vpp_plugin_dir = [
f"{config.vpp_install_dir}/vpp/{lib}/vpp_plugins" for lib in libs]
if config.vpp_test_plugin_dir is None:
config.vpp_test_plugin_dir = [
f"{config.vpp_install_dir}/vpp/{lib}/vpp_api_test_plugins"
for lib in libs]
test_dirs = [f"{ws}/test"]
if config.test_src_dir is not None:
test_dirs.extend(config.test_src_dir)
config.test_src_dir = test_dirs
if config.venv_dir is None:
config.venv_dir = f"{ws}/test/venv"
available_cpus = psutil.Process().cpu_affinity()
num_cpus = len(available_cpus)
if config.max_vpp_cpus == 'auto':
max_vpp_cpus = num_cpus
elif config.max_vpp_cpus > 0:
max_vpp_cpus = min(config.max_vpp_cpus, num_cpus)
else:
max_vpp_cpus = num_cpus
if __name__ == "__main__":
print("Provided arguments:")
for i in config.__dict__:
print(f" {i} is {config.__dict__[i]}")