oss-fuzz/infra/base-images/base-runner/python_coverage_runner_help.py

182 lines
5.6 KiB
Python
Executable File

# Copyright 2022 Google LLC
#
# Licensed under the Apache License, Version 2.0 (the "License");
# you may not use this file except in compliance with the License.
# You may obtain a copy of the License at
#
# http://www.apache.org/licenses/LICENSE-2.0
#
# Unless required by applicable law or agreed to in writing, software
# distributed under the License is distributed on an "AS IS" BASIS,
# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
# See the License for the specific language governing permissions and
# limitations under the License.
"""
Helper to manage coverage.py related operations. Does two main
things: (1) pass commands into the coverage.py library and (2)
translate .coverage created from a pyinstaller executable into
paths that match local files. This is needed for html report creation.
"""
import os
import re
import json
import sys
from coverage.cmdline import main as coverage_main
from coverage.data import CoverageData
def should_exclude_file(filepath):
"""Returns whether the path should be excluded from the coverage report."""
# Skip all atheris code
if "atheris" in filepath:
return True
# Filter out all standard python libraries
if ('/usr/local/lib/python' in filepath and
'site-packages' not in filepath and 'dist-packages' not in filepath):
return True
# Avoid all PyInstaller modules.
if 'PyInstaller' in filepath:
return True
return False
def translate_lines(cov_data, new_cov_data, all_file_paths):
"""
Translate lines in a .coverage file created by coverage.py such that
the file paths points to local files instead. This is needed when collecting
coverage from executables created by pyinstaller.
"""
for pyinstaller_file_path in cov_data.measured_files():
stripped_py_file_path = pyinstaller_file_path
if stripped_py_file_path.startswith('/tmp/_MEI'):
stripped_py_file_path = '/'.join(stripped_py_file_path.split('/')[3:])
if stripped_py_file_path.startswith('/out/'):
stripped_py_file_path = stripped_py_file_path.replace('/out/', '')
# Check if this file exists in our file paths:
for local_file_path in all_file_paths:
if should_exclude_file(local_file_path):
continue
if local_file_path.endswith(stripped_py_file_path):
print('Found matching: %s' % (local_file_path))
new_cov_data.add_lines(
{local_file_path: cov_data.lines(pyinstaller_file_path)})
def translate_coverage(all_file_paths):
"""
Translate pyinstaller-generated file paths in .coverage (produced by
coverage.py) into local file paths. Place result in .new_coverage.
"""
covdata_pre_translation = CoverageData('.coverage')
covdata_post_translation = CoverageData('.new_coverage')
covdata_pre_translation.read()
translate_lines(covdata_pre_translation, covdata_post_translation,
all_file_paths)
covdata_post_translation.write()
def convert_coveragepy_cov_to_summary_json(src, dst):
"""
Converts a json file produced by coveragepy into a summary.json file
similary to llvm-cov output. `src` is the source coveragepy json file,
`dst` is the destination json file, which will be overwritten.
"""
dst_dict = {'data': [{'files': {}}]}
lines_covered = 0
lines_count = 0
with open(src, "r") as src_f:
src_json = json.loads(src_f.read())
if 'files' in src_json:
for elem in src_json.get('files'):
if 'summary' not in src_json['files'][elem]:
continue
src_dict = src_json['files'][elem]['summary']
count = src_dict['covered_lines'] + src_dict['missing_lines']
covered = src_dict['covered_lines']
notcovered = src_dict['missing_lines']
percent = src_dict['percent_covered']
# Accumulate line coverage
lines_covered += covered
lines_count += count
dst_dict['data'][0]['files'][elem] = {
'summary': {
'lines': {
'count': count,
'covered': covered,
'notcovered': notcovered,
'percent': percent
}
}
}
if lines_count > 0:
lines_covered_percent = lines_covered / lines_count
else:
lines_covered_percent = 0.0
dst_dict['data'][0]['totals'] = {
'branches': {
'count': 0,
'covered': 0,
'notcovered': 0,
'percent': 0.0
},
'functions': {
'count': 0,
'covered': 0,
'percent': 0.0
},
'instantiations': {
'count': 0,
'covered': 0,
'percent': 0.0
},
'lines': {
'count': lines_count,
'covered': lines_covered,
'percent': lines_covered_percent
},
'regions': {
'count': 0,
'covered': 0,
'notcovered': 0,
'percent': 0.0
}
}
with open(dst, 'w') as dst_f:
dst_f.write(json.dumps(dst_dict))
def main():
"""
Main handler.
"""
if sys.argv[1] == 'translate':
print('Translating the coverage')
files_path = sys.argv[2]
all_file_paths = list()
for root, _, files in os.walk(files_path):
for relative_file_path in files:
abs_file_path = os.path.abspath(os.path.join(root, relative_file_path))
all_file_paths.append(abs_file_path)
print('Done with path walk')
translate_coverage(all_file_paths)
elif sys.argv[1] == 'convert-to-summary-json':
src = sys.argv[2]
dst = sys.argv[3]
convert_coveragepy_cov_to_summary_json(src, dst)
else:
# Pass commands into coverage package
sys.argv[0] = re.sub(r'(-script\.pyw|\.exe)?$', '', sys.argv[0])
sys.exit(coverage_main())
if __name__ == '__main__':
main()