Simon Glass | ff1fd6c | 2018-07-06 10:27:23 -0600 | [diff] [blame] | 1 | # SPDX-License-Identifier: GPL-2.0+ |
| 2 | # |
| 3 | # Copyright (c) 2016 Google, Inc |
| 4 | # |
| 5 | |
Simon Glass | c3f9454 | 2018-07-06 10:27:34 -0600 | [diff] [blame] | 6 | from contextlib import contextmanager |
Simon Glass | ff1fd6c | 2018-07-06 10:27:23 -0600 | [diff] [blame] | 7 | import glob |
| 8 | import os |
| 9 | import sys |
| 10 | |
| 11 | import command |
| 12 | |
Simon Glass | c3a13cc | 2020-04-17 18:08:55 -0600 | [diff] [blame^] | 13 | from io import StringIO |
Simon Glass | c3f9454 | 2018-07-06 10:27:34 -0600 | [diff] [blame] | 14 | |
Simon Glass | 9550f9a | 2019-05-17 22:00:54 -0600 | [diff] [blame] | 15 | PYTHON = 'python%d' % sys.version_info[0] |
| 16 | |
Simon Glass | c3f9454 | 2018-07-06 10:27:34 -0600 | [diff] [blame] | 17 | |
Simon Glass | ff1fd6c | 2018-07-06 10:27:23 -0600 | [diff] [blame] | 18 | def RunTestCoverage(prog, filter_fname, exclude_list, build_dir, required=None): |
| 19 | """Run tests and check that we get 100% coverage |
| 20 | |
| 21 | Args: |
| 22 | prog: Program to run (with be passed a '-t' argument to run tests |
| 23 | filter_fname: Normally all *.py files in the program's directory will |
| 24 | be included. If this is not None, then it is used to filter the |
| 25 | list so that only filenames that don't contain filter_fname are |
| 26 | included. |
| 27 | exclude_list: List of file patterns to exclude from the coverage |
| 28 | calculation |
| 29 | build_dir: Build directory, used to locate libfdt.py |
| 30 | required: List of modules which must be in the coverage report |
| 31 | |
| 32 | Raises: |
| 33 | ValueError if the code coverage is not 100% |
| 34 | """ |
| 35 | # This uses the build output from sandbox_spl to get _libfdt.so |
| 36 | path = os.path.dirname(prog) |
| 37 | if filter_fname: |
| 38 | glob_list = glob.glob(os.path.join(path, '*.py')) |
| 39 | glob_list = [fname for fname in glob_list if filter_fname in fname] |
| 40 | else: |
| 41 | glob_list = [] |
| 42 | glob_list += exclude_list |
Simon Glass | 9550f9a | 2019-05-17 22:00:54 -0600 | [diff] [blame] | 43 | glob_list += ['*libfdt.py', '*site-packages*', '*dist-packages*'] |
Simon Glass | 53cd5d9 | 2019-07-08 14:25:29 -0600 | [diff] [blame] | 44 | test_cmd = 'test' if 'binman.py' in prog else '-t' |
Simon Glass | 9550f9a | 2019-05-17 22:00:54 -0600 | [diff] [blame] | 45 | cmd = ('PYTHONPATH=$PYTHONPATH:%s/sandbox_spl/tools %s-coverage run ' |
Simon Glass | 53cd5d9 | 2019-07-08 14:25:29 -0600 | [diff] [blame] | 46 | '--omit "%s" %s %s -P1' % (build_dir, PYTHON, ','.join(glob_list), |
| 47 | prog, test_cmd)) |
Simon Glass | ff1fd6c | 2018-07-06 10:27:23 -0600 | [diff] [blame] | 48 | os.system(cmd) |
Simon Glass | 9550f9a | 2019-05-17 22:00:54 -0600 | [diff] [blame] | 49 | stdout = command.Output('%s-coverage' % PYTHON, 'report') |
Simon Glass | ff1fd6c | 2018-07-06 10:27:23 -0600 | [diff] [blame] | 50 | lines = stdout.splitlines() |
| 51 | if required: |
| 52 | # Convert '/path/to/name.py' just the module name 'name' |
| 53 | test_set = set([os.path.splitext(os.path.basename(line.split()[0]))[0] |
| 54 | for line in lines if '/etype/' in line]) |
| 55 | missing_list = required |
Simon Glass | e430440 | 2019-07-08 14:25:32 -0600 | [diff] [blame] | 56 | missing_list.discard('__init__') |
Simon Glass | ff1fd6c | 2018-07-06 10:27:23 -0600 | [diff] [blame] | 57 | missing_list.difference_update(test_set) |
| 58 | if missing_list: |
Simon Glass | 5a1af1d | 2019-05-14 15:53:36 -0600 | [diff] [blame] | 59 | print('Missing tests for %s' % (', '.join(missing_list))) |
| 60 | print(stdout) |
Simon Glass | ff1fd6c | 2018-07-06 10:27:23 -0600 | [diff] [blame] | 61 | ok = False |
| 62 | |
| 63 | coverage = lines[-1].split(' ')[-1] |
| 64 | ok = True |
Simon Glass | 5a1af1d | 2019-05-14 15:53:36 -0600 | [diff] [blame] | 65 | print(coverage) |
Simon Glass | ff1fd6c | 2018-07-06 10:27:23 -0600 | [diff] [blame] | 66 | if coverage != '100%': |
Simon Glass | 5a1af1d | 2019-05-14 15:53:36 -0600 | [diff] [blame] | 67 | print(stdout) |
Simon Glass | 9550f9a | 2019-05-17 22:00:54 -0600 | [diff] [blame] | 68 | print("Type '%s-coverage html' to get a report in " |
| 69 | 'htmlcov/index.html' % PYTHON) |
Simon Glass | 5a1af1d | 2019-05-14 15:53:36 -0600 | [diff] [blame] | 70 | print('Coverage error: %s, but should be 100%%' % coverage) |
Simon Glass | ff1fd6c | 2018-07-06 10:27:23 -0600 | [diff] [blame] | 71 | ok = False |
| 72 | if not ok: |
| 73 | raise ValueError('Test coverage failure') |
Simon Glass | c3f9454 | 2018-07-06 10:27:34 -0600 | [diff] [blame] | 74 | |
| 75 | |
| 76 | # Use this to suppress stdout/stderr output: |
| 77 | # with capture_sys_output() as (stdout, stderr) |
| 78 | # ...do something... |
| 79 | @contextmanager |
| 80 | def capture_sys_output(): |
| 81 | capture_out, capture_err = StringIO(), StringIO() |
| 82 | old_out, old_err = sys.stdout, sys.stderr |
| 83 | try: |
| 84 | sys.stdout, sys.stderr = capture_out, capture_err |
| 85 | yield capture_out, capture_err |
| 86 | finally: |
| 87 | sys.stdout, sys.stderr = old_out, old_err |