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 | 5a1af1d | 2019-05-14 15:53:36 -0600 | [diff] [blame] | 6 | from __future__ import print_function |
| 7 | |
Simon Glass | c3f9454 | 2018-07-06 10:27:34 -0600 | [diff] [blame] | 8 | from contextlib import contextmanager |
Simon Glass | ff1fd6c | 2018-07-06 10:27:23 -0600 | [diff] [blame] | 9 | import glob |
| 10 | import os |
| 11 | import sys |
| 12 | |
| 13 | import command |
| 14 | |
Simon Glass | c3f9454 | 2018-07-06 10:27:34 -0600 | [diff] [blame] | 15 | try: |
| 16 | from StringIO import StringIO |
| 17 | except ImportError: |
| 18 | from io import StringIO |
| 19 | |
| 20 | |
Simon Glass | ff1fd6c | 2018-07-06 10:27:23 -0600 | [diff] [blame] | 21 | def RunTestCoverage(prog, filter_fname, exclude_list, build_dir, required=None): |
| 22 | """Run tests and check that we get 100% coverage |
| 23 | |
| 24 | Args: |
| 25 | prog: Program to run (with be passed a '-t' argument to run tests |
| 26 | filter_fname: Normally all *.py files in the program's directory will |
| 27 | be included. If this is not None, then it is used to filter the |
| 28 | list so that only filenames that don't contain filter_fname are |
| 29 | included. |
| 30 | exclude_list: List of file patterns to exclude from the coverage |
| 31 | calculation |
| 32 | build_dir: Build directory, used to locate libfdt.py |
| 33 | required: List of modules which must be in the coverage report |
| 34 | |
| 35 | Raises: |
| 36 | ValueError if the code coverage is not 100% |
| 37 | """ |
| 38 | # This uses the build output from sandbox_spl to get _libfdt.so |
| 39 | path = os.path.dirname(prog) |
| 40 | if filter_fname: |
| 41 | glob_list = glob.glob(os.path.join(path, '*.py')) |
| 42 | glob_list = [fname for fname in glob_list if filter_fname in fname] |
| 43 | else: |
| 44 | glob_list = [] |
| 45 | glob_list += exclude_list |
| 46 | glob_list += ['*libfdt.py', '*site-packages*'] |
| 47 | cmd = ('PYTHONPATH=$PYTHONPATH:%s/sandbox_spl/tools python-coverage run ' |
Simon Glass | 11ae93e | 2018-10-01 21:12:47 -0600 | [diff] [blame] | 48 | '--omit "%s" %s -P1 -t' % (build_dir, ','.join(glob_list), prog)) |
Simon Glass | ff1fd6c | 2018-07-06 10:27:23 -0600 | [diff] [blame] | 49 | os.system(cmd) |
| 50 | stdout = command.Output('python-coverage', 'report') |
| 51 | lines = stdout.splitlines() |
| 52 | if required: |
| 53 | # Convert '/path/to/name.py' just the module name 'name' |
| 54 | test_set = set([os.path.splitext(os.path.basename(line.split()[0]))[0] |
| 55 | for line in lines if '/etype/' in line]) |
| 56 | missing_list = required |
| 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) |
| 68 | print("Type 'python-coverage html' to get a report in " |
| 69 | 'htmlcov/index.html') |
| 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 |