blob: 8555cdff9aa9f3dd25202e29701644b02a8a0a21 [file] [log] [blame]
Mike Frysingerf1ba7ad2022-09-12 05:42:57 -04001# Copyright 2020 The ChromiumOS Authors
Chris McDonald17d86b32020-03-18 17:28:43 -06002# Use of this source code is governed by a BSD-style license that can be
3# found in the LICENSE file.
4
Mike Frysinger4975e5a2021-04-13 17:06:09 -04005"""Chromite main test runner.
6
7Run the specified tests. If none are specified, we'll scan the
8tree looking for tests to run and then only run the semi-fast ones.
Mike Frysinger2aa424f2021-07-19 21:37:29 -04009
Brian Norrisd7d91502023-08-18 16:39:38 -070010https://docs.pytest.org/en/latest/how-to/usage.html#specifying-which-tests-to-run
Mike Frysinger2aa424f2021-07-19 21:37:29 -040011
12Examples:
13# Run all tests in a module.
14$ ./run_tests lib/osutils_unittest.py
15# Run a class of tests in a module.
16$ ./run_tests lib/osutils_unittest.py::TestOsutils
17# Run a single test.
18$ ./run_tests lib/osutils_unittest.py::TestOsutils::testIsSubPath
19# List all tests that'd be run.
20$ ./run_tests -- --collect-only
Mike Frysinger4975e5a2021-04-13 17:06:09 -040021"""
Chris McDonald17d86b32020-03-18 17:28:43 -060022
Chris McDonald59650c32021-07-20 15:29:28 -060023import logging
Chris McDonald3c557392020-03-31 13:41:46 -060024import os
Chris McDonald17d86b32020-03-18 17:28:43 -060025import sys
26
27import pytest # pylint: disable=import-error
28
Alex Klein9ea00b52021-02-24 14:47:22 -070029from chromite.api import compile_build_api_proto
Mike Frysingerfbb1c142023-03-10 00:47:21 -050030from chromite.format import formatters
Mike Frysingerd7af8462020-11-11 03:44:54 -050031from chromite.lib import commandline
Chris McDonald3c557392020-03-31 13:41:46 -060032from chromite.lib import constants
33from chromite.lib import cros_build_lib
34from chromite.lib import gs
35from chromite.lib import namespaces
Mike Frysinger3763bf22023-03-30 12:38:28 -040036from chromite.lint import linters
Mike Frysingerfbb1c142023-03-10 00:47:21 -050037from chromite.scripts import clang_format
Chris McDonald3c557392020-03-31 13:41:46 -060038
Chris McDonald17d86b32020-03-18 17:28:43 -060039
40def main(argv):
Alex Klein1699fab2022-09-08 08:46:06 -060041 parser = get_parser()
42 opts = parser.parse_args()
43 opts.Freeze()
Mike Frysingerd7af8462020-11-11 03:44:54 -050044
Alex Klein1699fab2022-09-08 08:46:06 -060045 pytest_args = opts.pytest_args
Mike Frysingerd7af8462020-11-11 03:44:54 -050046
Alex Klein1699fab2022-09-08 08:46:06 -060047 if opts.chroot:
48 ensure_chroot_exists()
49 re_execute_inside_chroot(argv)
50 else:
51 pytest_args += ["--no-chroot"]
Chris McDonald3c557392020-03-31 13:41:46 -060052
Alex Klein1699fab2022-09-08 08:46:06 -060053 if opts.network:
54 pytest_args += ["-m", "not network_test or network_test"]
Mike Frysingerc73c6b82021-04-22 00:07:54 -040055
Mike Frysinger08b10122023-03-29 21:46:50 -040056 if opts.precache:
57 precache()
Chris McDonald3c557392020-03-31 13:41:46 -060058
Alex Klein1699fab2022-09-08 08:46:06 -060059 if opts.quick:
60 logging.info("Skipping test namespacing due to --quickstart.")
Alex Klein1699fab2022-09-08 08:46:06 -060061 else:
Trent Apted4a0812b2023-05-15 15:33:55 +100062 # Namespacing is enabled by default because tests may break each other
63 # or interfere with parts of the running system if not isolated in a
64 # namespace. Disabling namespaces is not recommended for general use.
Alex Klein1699fab2022-09-08 08:46:06 -060065 namespaces.ReExecuteWithNamespace(
Mike Frysinger08b10122023-03-29 21:46:50 -040066 [sys.argv[0], "--no-precache"] + argv, network=opts.network
Alex Klein1699fab2022-09-08 08:46:06 -060067 )
Chris McDonald3c557392020-03-31 13:41:46 -060068
Mike Frysinger55aa3092023-02-15 09:43:22 -050069 jobs = opts.jobs
70 if jobs is None:
Trent Apted4a0812b2023-05-15 15:33:55 +100071 # Default to running in a single process under --quickstart. User args
Alex Kleinde659352023-10-03 12:14:21 -060072 # can still override this. Cap it at 64 by default to prevent the
73 # overhead from spawning too many nodes.
74 jobs = 0 if opts.quick else min(os.cpu_count(), 64)
Mike Frysinger55aa3092023-02-15 09:43:22 -050075 pytest_args = ["-n", str(jobs)] + pytest_args
76
Alex Klein1699fab2022-09-08 08:46:06 -060077 # Check the environment. https://crbug.com/1015450
78 st = os.stat("/")
79 if st.st_mode & 0o007 != 0o005:
80 cros_build_lib.Die(
81 f"The root directory has broken permissions: {st.st_mode:o}\n"
82 "Fix with: sudo chmod o+rx-w /"
83 )
84 if st.st_uid or st.st_gid:
85 cros_build_lib.Die(
86 f"The root directory has broken ownership: {st.st_uid}:{st.st_gid}"
87 " (should be 0:0)\nFix with: sudo chown 0:0 /"
88 )
Mike Frysinger55919ef2021-04-01 00:55:39 -040089
Alex Klein1699fab2022-09-08 08:46:06 -060090 logging.debug("Running: pytest %s", cros_build_lib.CmdToStr(pytest_args))
91 sys.exit(pytest.main(pytest_args))
Chris McDonald3c557392020-03-31 13:41:46 -060092
93
Alex Klein9ea00b52021-02-24 14:47:22 -070094def precache():
Alex Klein1699fab2022-09-08 08:46:06 -060095 """Do some network-dependent stuff before we disallow network access."""
Mike Frysingerfbb1c142023-03-10 00:47:21 -050096 # pylint: disable=protected-access
Mike Frysinger0aa0d582023-03-24 12:55:01 -040097 logging.notice("Caching tools from network (cipd/vpython/etc...)")
Mike Frysingerfbb1c142023-03-10 00:47:21 -050098
Alex Klein1699fab2022-09-08 08:46:06 -060099 # This is a cheesy hack to make sure gsutil is populated in the cache before
100 # we run tests. This is a partial workaround for crbug.com/468838.
101 gs.GSContext.InitializeCache()
102 # Ensure protoc is installed for api/compile_build_api_proto_unittest.
103 compile_build_api_proto.InstallProtoc(
104 compile_build_api_proto.ProtocVersion.CHROMITE
105 )
Mike Frysingerfbb1c142023-03-10 00:47:21 -0500106 # Ensure various tools are available.
Mike Frysinger29e1da92023-03-21 10:52:43 -0400107 cros_build_lib.dbg_run(
Mike Frysingera69df982023-03-21 16:52:27 -0400108 [constants.CHROMITE_DIR / "scripts" / "black", "--version"],
Mike Frysinger29e1da92023-03-21 10:52:43 -0400109 capture_output=True,
110 )
111 cros_build_lib.dbg_run(
Mike Frysingera69df982023-03-21 16:52:27 -0400112 [constants.CHROMITE_DIR / "scripts" / "isort", "--version"],
Mike Frysinger29e1da92023-03-21 10:52:43 -0400113 capture_output=True,
114 )
Mike Frysinger0aa0d582023-03-24 12:55:01 -0400115 formatters.gn._find_gn()
Mike Frysingerfbb1c142023-03-10 00:47:21 -0500116 formatters.star._find_buildifier()
117 formatters.textproto._find_txtpbfmt()
Mike Frysinger3763bf22023-03-30 12:38:28 -0400118 linters.shell._find_shellcheck()
Mike Frysingerfbb1c142023-03-10 00:47:21 -0500119 with clang_format.ClangFormat():
120 pass
Alex Klein9ea00b52021-02-24 14:47:22 -0700121
122
Chris McDonald3c557392020-03-31 13:41:46 -0600123def re_execute_inside_chroot(argv):
Alex Klein1699fab2022-09-08 08:46:06 -0600124 """Re-execute the test wrapper inside the chroot."""
125 if cros_build_lib.IsInsideChroot():
126 return
Mike Frysinger6df594e2020-11-11 15:02:59 -0500127
Mike Frysingera69df982023-03-21 16:52:27 -0400128 target = constants.CHROMITE_DIR / "scripts" / "run_tests"
Alex Klein1699fab2022-09-08 08:46:06 -0600129 relpath = os.path.relpath(target, ".")
130 # If we're in the scripts dir, make sure we always have a relative path,
131 # otherwise cros_sdk will search $PATH and fail.
132 if os.path.sep not in relpath:
133 relpath = os.path.join(".", relpath)
134 cmd = [
135 "cros_sdk",
136 "--working-dir",
137 ".",
138 "--",
139 relpath,
140 ]
141 os.execvp(cmd[0], cmd + argv)
Chris McDonald3c557392020-03-31 13:41:46 -0600142
143
144def ensure_chroot_exists():
Alex Klein1699fab2022-09-08 08:46:06 -0600145 """Ensure that a chroot exists for us to run tests in."""
146 chroot = os.path.join(constants.SOURCE_ROOT, constants.DEFAULT_CHROOT_DIR)
147 if not os.path.exists(chroot) and not cros_build_lib.IsInsideChroot():
148 cros_build_lib.run(["cros_sdk", "--create"])
Chris McDonald5d1af6a2020-04-21 08:00:15 -0600149
150
151def get_parser():
Alex Klein1699fab2022-09-08 08:46:06 -0600152 """Build the parser for command line arguments."""
153 parser = commandline.ArgumentParser(
154 description=__doc__,
155 epilog="To see the help output for pytest:\n$ %(prog)s -- --help",
Mike Frysinger9abf4b52023-09-15 11:22:43 -0400156 default_log_level="notice",
Alex Klein1699fab2022-09-08 08:46:06 -0600157 )
158 parser.add_argument(
Mike Frysinger55aa3092023-02-15 09:43:22 -0500159 "-j",
160 "--jobs",
161 type=int,
162 default=None,
163 help="Number of tests to run in parallel.",
164 )
165 parser.add_argument(
Alex Klein1699fab2022-09-08 08:46:06 -0600166 "--quickstart",
167 dest="quick",
168 action="store_true",
Trent Aptedc20bb6d2023-05-10 15:00:03 +1000169 help=(
170 "Skip normal test sandboxing and namespacing for faster start up "
171 "time."
172 ),
Alex Klein1699fab2022-09-08 08:46:06 -0600173 )
174 parser.add_argument(
175 "--network",
176 action="store_true",
177 help="Include network tests.",
178 )
179 parser.add_argument(
Mike Frysinger08b10122023-03-29 21:46:50 -0400180 "--no-precache",
181 dest="precache",
182 action="store_false",
183 help="Skip precaching packages from the network.",
184 )
185 parser.add_argument(
Alex Klein1699fab2022-09-08 08:46:06 -0600186 "--no-chroot",
187 dest="chroot",
188 action="store_false",
Trent Aptedc20bb6d2023-05-10 15:00:03 +1000189 help=(
190 "Don't initialize or enter a chroot for the test invocation. May "
191 "cause tests to unexpectedly fail!"
192 ),
Alex Klein1699fab2022-09-08 08:46:06 -0600193 )
194 parser.add_argument(
195 "pytest_args",
196 metavar="pytest arguments",
197 nargs="*",
198 help="Arguments to pass down to pytest (use -- to help separate)",
199 )
200 return parser