Chris McDonald | 17d86b3 | 2020-03-18 17:28:43 -0600 | [diff] [blame] | 1 | # Copyright 2020 The Chromium OS Authors. All rights reserved. |
| 2 | # Use of this source code is governed by a BSD-style license that can be |
| 3 | # found in the LICENSE file. |
| 4 | |
Mike Frysinger | 4975e5a | 2021-04-13 17:06:09 -0400 | [diff] [blame] | 5 | """Chromite main test runner. |
| 6 | |
| 7 | Run the specified tests. If none are specified, we'll scan the |
| 8 | tree looking for tests to run and then only run the semi-fast ones. |
Mike Frysinger | 2aa424f | 2021-07-19 21:37:29 -0400 | [diff] [blame] | 9 | |
| 10 | https://docs.pytest.org/en/latest/how-to/usage.html#specifying-tests-selecting-tests |
| 11 | |
| 12 | Examples: |
| 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 Frysinger | 4975e5a | 2021-04-13 17:06:09 -0400 | [diff] [blame] | 21 | """ |
Chris McDonald | 17d86b3 | 2020-03-18 17:28:43 -0600 | [diff] [blame] | 22 | |
Chris McDonald | 59650c3 | 2021-07-20 15:29:28 -0600 | [diff] [blame] | 23 | import logging |
Chris McDonald | 3c55739 | 2020-03-31 13:41:46 -0600 | [diff] [blame] | 24 | import os |
Chris McDonald | 17d86b3 | 2020-03-18 17:28:43 -0600 | [diff] [blame] | 25 | import sys |
| 26 | |
| 27 | import pytest # pylint: disable=import-error |
| 28 | |
Mike Frysinger | d7af846 | 2020-11-11 03:44:54 -0500 | [diff] [blame] | 29 | from chromite.lib import commandline |
Chris McDonald | 3c55739 | 2020-03-31 13:41:46 -0600 | [diff] [blame] | 30 | from chromite.lib import constants |
| 31 | from chromite.lib import cros_build_lib |
| 32 | from chromite.lib import gs |
| 33 | from chromite.lib import namespaces |
Ram Chandrasekar | 6975128 | 2022-02-25 21:07:36 +0000 | [diff] [blame] | 34 | from chromite.lib import osutils |
Chris McDonald | 3c55739 | 2020-03-31 13:41:46 -0600 | [diff] [blame] | 35 | |
Chris McDonald | 17d86b3 | 2020-03-18 17:28:43 -0600 | [diff] [blame] | 36 | |
| 37 | def main(argv): |
Chris McDonald | 5d1af6a | 2020-04-21 08:00:15 -0600 | [diff] [blame] | 38 | parser = get_parser() |
Mike Frysinger | d7af846 | 2020-11-11 03:44:54 -0500 | [diff] [blame] | 39 | opts = parser.parse_args() |
Mike Frysinger | a819cb3 | 2021-04-01 00:55:10 -0400 | [diff] [blame] | 40 | opts.Freeze() |
Mike Frysinger | d7af846 | 2020-11-11 03:44:54 -0500 | [diff] [blame] | 41 | |
| 42 | pytest_args = opts.pytest_args |
| 43 | |
Chris McDonald | 5d1af6a | 2020-04-21 08:00:15 -0600 | [diff] [blame] | 44 | if opts.quick: |
Chris McDonald | 653510e | 2020-05-01 17:15:16 -0600 | [diff] [blame] | 45 | if not cros_build_lib.IsInsideChroot() and opts.chroot: |
Mike Frysinger | 968c114 | 2020-05-09 00:37:56 -0400 | [diff] [blame] | 46 | logging.warning('Tests start up faster when run from inside the chroot.') |
Chris McDonald | 5d1af6a | 2020-04-21 08:00:15 -0600 | [diff] [blame] | 47 | |
Chris McDonald | 653510e | 2020-05-01 17:15:16 -0600 | [diff] [blame] | 48 | if opts.chroot: |
| 49 | ensure_chroot_exists() |
| 50 | re_execute_inside_chroot(argv) |
| 51 | else: |
Chris McDonald | cdfd113 | 2020-05-12 07:09:51 -0600 | [diff] [blame] | 52 | pytest_args += ['--no-chroot'] |
Chris McDonald | 3c55739 | 2020-03-31 13:41:46 -0600 | [diff] [blame] | 53 | |
Mike Frysinger | c73c6b8 | 2021-04-22 00:07:54 -0400 | [diff] [blame] | 54 | if opts.network: |
| 55 | pytest_args += ['-m', 'not network_test or network_test'] |
| 56 | |
Chris McDonald | 3c55739 | 2020-03-31 13:41:46 -0600 | [diff] [blame] | 57 | # This is a cheesy hack to make sure gsutil is populated in the cache before |
| 58 | # we run tests. This is a partial workaround for crbug.com/468838. |
Mike Frysinger | 682b333 | 2021-06-22 11:07:17 -0400 | [diff] [blame] | 59 | gs.GSContext.InitializeCache() |
Chris McDonald | 3c55739 | 2020-03-31 13:41:46 -0600 | [diff] [blame] | 60 | |
Chris McDonald | 5d1af6a | 2020-04-21 08:00:15 -0600 | [diff] [blame] | 61 | if opts.quick: |
| 62 | logging.info('Skipping test namespacing due to --quickstart.') |
| 63 | # Default to running in a single process under --quickstart. User args can |
| 64 | # still override this. |
| 65 | pytest_args = ['-n', '0'] + pytest_args |
| 66 | else: |
| 67 | # Namespacing is enabled by default because tests may break each other or |
| 68 | # interfere with parts of the running system if not isolated in a namespace. |
| 69 | # Disabling namespaces is not recommended for general use. |
Mike Frysinger | c73c6b8 | 2021-04-22 00:07:54 -0400 | [diff] [blame] | 70 | re_execute_with_namespace([sys.argv[0]] + argv, network=opts.network) |
Chris McDonald | 3c55739 | 2020-03-31 13:41:46 -0600 | [diff] [blame] | 71 | |
Mike Frysinger | 55919ef | 2021-04-01 00:55:39 -0400 | [diff] [blame] | 72 | # Check the environment. https://crbug.com/1015450 |
| 73 | st = os.stat('/') |
Henrique Ferreiro | 41ba299 | 2022-04-28 14:52:06 +0200 | [diff] [blame^] | 74 | if st.st_mode & 0o007 != 0o005: |
Mike Frysinger | 55919ef | 2021-04-01 00:55:39 -0400 | [diff] [blame] | 75 | cros_build_lib.Die( |
| 76 | f'The root directory has broken permissions: {st.st_mode:o}\n' |
Henrique Ferreiro | 41ba299 | 2022-04-28 14:52:06 +0200 | [diff] [blame^] | 77 | 'Fix with: sudo chmod o+rx-w /') |
Mike Frysinger | 55919ef | 2021-04-01 00:55:39 -0400 | [diff] [blame] | 78 | if st.st_uid or st.st_gid: |
| 79 | cros_build_lib.Die( |
| 80 | f'The root directory has broken ownership: {st.st_uid}:{st.st_gid}' |
| 81 | ' (should be 0:0)\nFix with: sudo chown 0:0 /') |
| 82 | |
Chris McDonald | 5d1af6a | 2020-04-21 08:00:15 -0600 | [diff] [blame] | 83 | sys.exit(pytest.main(pytest_args)) |
Chris McDonald | 3c55739 | 2020-03-31 13:41:46 -0600 | [diff] [blame] | 84 | |
| 85 | |
| 86 | def re_execute_with_namespace(argv, network=False): |
| 87 | """Re-execute as root so we can unshare resources.""" |
Ram Chandrasekar | 6975128 | 2022-02-25 21:07:36 +0000 | [diff] [blame] | 88 | if osutils.IsNonRootUser(): |
Chris McDonald | 3c55739 | 2020-03-31 13:41:46 -0600 | [diff] [blame] | 89 | cmd = [ |
| 90 | 'sudo', |
| 91 | 'HOME=%s' % os.environ['HOME'], |
| 92 | 'PATH=%s' % os.environ['PATH'], |
| 93 | '--', |
| 94 | ] + argv |
| 95 | os.execvp(cmd[0], cmd) |
| 96 | else: |
Chris McDonald | 3c55739 | 2020-03-31 13:41:46 -0600 | [diff] [blame] | 97 | namespaces.SimpleUnshare(net=not network, pid=True) |
| 98 | # We got our namespaces, so switch back to the user to run the tests. |
| 99 | gid = int(os.environ.pop('SUDO_GID')) |
| 100 | uid = int(os.environ.pop('SUDO_UID')) |
| 101 | user = os.environ.pop('SUDO_USER') |
| 102 | os.initgroups(user, gid) |
| 103 | os.setresgid(gid, gid, gid) |
| 104 | os.setresuid(uid, uid, uid) |
| 105 | os.environ['USER'] = user |
| 106 | |
| 107 | |
| 108 | def re_execute_inside_chroot(argv): |
| 109 | """Re-execute the test wrapper inside the chroot.""" |
Mike Frysinger | 6df594e | 2020-11-11 15:02:59 -0500 | [diff] [blame] | 110 | if cros_build_lib.IsInsideChroot(): |
| 111 | return |
| 112 | |
Mike Frysinger | 4975e5a | 2021-04-13 17:06:09 -0400 | [diff] [blame] | 113 | target = os.path.join(constants.CHROMITE_DIR, 'scripts', 'run_tests') |
Mike Frysinger | 6df594e | 2020-11-11 15:02:59 -0500 | [diff] [blame] | 114 | relpath = os.path.relpath(target, '.') |
| 115 | # If we're in the scripts dir, make sure we always have a relative path, |
| 116 | # otherwise cros_sdk will search $PATH and fail. |
| 117 | if os.path.sep not in relpath: |
| 118 | relpath = os.path.join('.', relpath) |
Chris McDonald | 3c55739 | 2020-03-31 13:41:46 -0600 | [diff] [blame] | 119 | cmd = [ |
| 120 | 'cros_sdk', |
Mike Frysinger | 6df594e | 2020-11-11 15:02:59 -0500 | [diff] [blame] | 121 | '--working-dir', '.', |
Chris McDonald | 3c55739 | 2020-03-31 13:41:46 -0600 | [diff] [blame] | 122 | '--', |
Mike Frysinger | 6df594e | 2020-11-11 15:02:59 -0500 | [diff] [blame] | 123 | relpath, |
Chris McDonald | 3c55739 | 2020-03-31 13:41:46 -0600 | [diff] [blame] | 124 | ] |
Mike Frysinger | 6df594e | 2020-11-11 15:02:59 -0500 | [diff] [blame] | 125 | os.execvp(cmd[0], cmd + argv) |
Chris McDonald | 3c55739 | 2020-03-31 13:41:46 -0600 | [diff] [blame] | 126 | |
| 127 | |
| 128 | def ensure_chroot_exists(): |
| 129 | """Ensure that a chroot exists for us to run tests in.""" |
| 130 | chroot = os.path.join(constants.SOURCE_ROOT, constants.DEFAULT_CHROOT_DIR) |
| 131 | if not os.path.exists(chroot) and not cros_build_lib.IsInsideChroot(): |
| 132 | cros_build_lib.run(['cros_sdk', '--create']) |
Chris McDonald | 5d1af6a | 2020-04-21 08:00:15 -0600 | [diff] [blame] | 133 | |
| 134 | |
| 135 | def get_parser(): |
| 136 | """Build the parser for command line arguments.""" |
Mike Frysinger | d7af846 | 2020-11-11 03:44:54 -0500 | [diff] [blame] | 137 | parser = commandline.ArgumentParser( |
Chris McDonald | 5d1af6a | 2020-04-21 08:00:15 -0600 | [diff] [blame] | 138 | description=__doc__, |
Mike Frysinger | d7af846 | 2020-11-11 03:44:54 -0500 | [diff] [blame] | 139 | epilog='To see the help output for pytest:\n$ %(prog)s -- --help', |
Chris McDonald | 5d1af6a | 2020-04-21 08:00:15 -0600 | [diff] [blame] | 140 | ) |
| 141 | parser.add_argument( |
| 142 | '--quickstart', |
| 143 | dest='quick', |
| 144 | action='store_true', |
| 145 | help='Skip normal test sandboxing and namespacing for faster start up ' |
| 146 | 'time.', |
| 147 | ) |
Chris McDonald | 653510e | 2020-05-01 17:15:16 -0600 | [diff] [blame] | 148 | parser.add_argument( |
Mike Frysinger | c73c6b8 | 2021-04-22 00:07:54 -0400 | [diff] [blame] | 149 | '--network', |
| 150 | action='store_true', |
| 151 | help='Include network tests.', |
| 152 | ) |
| 153 | parser.add_argument( |
Chris McDonald | 653510e | 2020-05-01 17:15:16 -0600 | [diff] [blame] | 154 | '--no-chroot', |
| 155 | dest='chroot', |
| 156 | action='store_false', |
| 157 | help="Don't initialize or enter a chroot for the test invocation. May " |
| 158 | 'cause tests to unexpectedly fail!', |
| 159 | ) |
Mike Frysinger | d7af846 | 2020-11-11 03:44:54 -0500 | [diff] [blame] | 160 | parser.add_argument( |
| 161 | 'pytest_args', |
| 162 | metavar='pytest arguments', |
| 163 | nargs='*', |
| 164 | help='Arguments to pass down to pytest (use -- to help separate)', |
| 165 | ) |
Chris McDonald | 5d1af6a | 2020-04-21 08:00:15 -0600 | [diff] [blame] | 166 | return parser |