2021-01-20 15:38:05 +00:00
|
|
|
# Copyright 2021 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.
|
|
|
|
"""Tests for coverage.py"""
|
2021-01-20 16:22:24 +00:00
|
|
|
import os
|
|
|
|
import json
|
2021-01-20 15:38:05 +00:00
|
|
|
import unittest
|
|
|
|
from unittest import mock
|
|
|
|
|
|
|
|
import coverage
|
|
|
|
|
2021-01-20 15:59:32 +00:00
|
|
|
# pylint: disable=protected-access
|
2021-01-20 15:38:05 +00:00
|
|
|
|
2021-03-19 18:49:38 +00:00
|
|
|
TEST_DATA_PATH = os.path.join(os.path.dirname(os.path.abspath(__file__)),
|
|
|
|
'test_data')
|
2021-01-20 16:22:24 +00:00
|
|
|
|
2021-01-20 19:19:15 +00:00
|
|
|
PROJECT_NAME = 'curl'
|
|
|
|
REPO_PATH = '/src/curl'
|
2021-01-20 19:24:25 +00:00
|
|
|
FUZZ_TARGET = 'curl_fuzzer'
|
2021-01-20 19:19:15 +00:00
|
|
|
PROJECT_COV_JSON_FILENAME = 'example_curl_cov.json'
|
2021-01-20 19:24:25 +00:00
|
|
|
FUZZ_TARGET_COV_JSON_FILENAME = 'example_curl_fuzzer_cov.json'
|
|
|
|
INVALID_TARGET = 'not-a-fuzz-target'
|
2021-01-20 19:19:15 +00:00
|
|
|
|
2021-03-19 18:49:38 +00:00
|
|
|
with open(os.path.join(TEST_DATA_PATH,
|
2021-01-20 19:19:15 +00:00
|
|
|
PROJECT_COV_JSON_FILENAME),) as cov_file_handle:
|
|
|
|
PROJECT_COV_INFO = json.loads(cov_file_handle.read())
|
|
|
|
|
2021-01-20 16:46:38 +00:00
|
|
|
|
2021-01-20 15:38:05 +00:00
|
|
|
class GetFuzzerStatsDirUrlTest(unittest.TestCase):
|
|
|
|
"""Tests _get_fuzzer_stats_dir_url."""
|
|
|
|
|
2021-03-03 22:46:51 +00:00
|
|
|
@mock.patch('coverage.get_json_from_url',
|
|
|
|
return_value={
|
|
|
|
'fuzzer_stats_dir':
|
|
|
|
'gs://oss-fuzz-coverage/systemd/fuzzer_stats/20210303'
|
|
|
|
})
|
2021-01-20 15:38:05 +00:00
|
|
|
def test_get_valid_project(self, mocked_get_json_from_url):
|
|
|
|
"""Tests that a project's coverage report can be downloaded and parsed.
|
|
|
|
|
2021-01-20 19:19:15 +00:00
|
|
|
NOTE: This test relies on the PROJECT_NAME repo's coverage report.
|
2021-01-20 15:38:05 +00:00
|
|
|
The "example" project was not used because it has no coverage reports.
|
|
|
|
"""
|
2021-03-03 22:46:51 +00:00
|
|
|
result = coverage._get_fuzzer_stats_dir_url(PROJECT_NAME)
|
2021-01-20 15:38:05 +00:00
|
|
|
(url,), _ = mocked_get_json_from_url.call_args
|
|
|
|
self.assertEqual(
|
|
|
|
'https://storage.googleapis.com/oss-fuzz-coverage/'
|
|
|
|
'latest_report_info/curl.json', url)
|
|
|
|
|
2021-03-03 22:46:51 +00:00
|
|
|
expected_result = (
|
|
|
|
'https://storage.googleapis.com/oss-fuzz-coverage/systemd/fuzzer_stats/'
|
|
|
|
'20210303')
|
|
|
|
self.assertEqual(result, expected_result)
|
|
|
|
|
2021-01-20 15:38:05 +00:00
|
|
|
def test_get_invalid_project(self):
|
|
|
|
"""Tests that passing a bad project returns None."""
|
|
|
|
self.assertIsNone(coverage._get_fuzzer_stats_dir_url('not-a-proj'))
|
|
|
|
|
|
|
|
|
2021-01-20 16:22:24 +00:00
|
|
|
class GetTargetCoverageReportTest(unittest.TestCase):
|
|
|
|
"""Tests get_target_coverage_report."""
|
|
|
|
|
|
|
|
def setUp(self):
|
|
|
|
with mock.patch('coverage._get_latest_cov_report_info',
|
2021-01-20 19:19:15 +00:00
|
|
|
return_value=PROJECT_COV_INFO):
|
2021-01-20 16:22:24 +00:00
|
|
|
self.coverage_getter = coverage.OssFuzzCoverageGetter(
|
2021-01-20 19:19:15 +00:00
|
|
|
PROJECT_NAME, REPO_PATH)
|
2021-01-20 16:22:24 +00:00
|
|
|
|
|
|
|
@mock.patch('coverage.get_json_from_url', return_value={})
|
|
|
|
def test_valid_target(self, mocked_get_json_from_url):
|
|
|
|
"""Tests that a target's coverage report can be downloaded and parsed."""
|
2021-01-20 19:24:25 +00:00
|
|
|
self.coverage_getter.get_target_coverage_report(FUZZ_TARGET)
|
2021-01-20 16:22:24 +00:00
|
|
|
(url,), _ = mocked_get_json_from_url.call_args
|
|
|
|
self.assertEqual(
|
|
|
|
'https://storage.googleapis.com/oss-fuzz-coverage/'
|
|
|
|
'curl/fuzzer_stats/20200226/curl_fuzzer.json', url)
|
|
|
|
|
|
|
|
def test_invalid_target(self):
|
|
|
|
"""Tests that passing an invalid target coverage report returns None."""
|
|
|
|
self.assertIsNone(
|
2021-01-20 19:24:25 +00:00
|
|
|
self.coverage_getter.get_target_coverage_report(INVALID_TARGET))
|
|
|
|
|
2021-02-03 23:18:56 +00:00
|
|
|
@mock.patch('coverage._get_latest_cov_report_info', return_value=None)
|
|
|
|
def test_invalid_project_json(self, _):
|
2021-01-20 19:24:25 +00:00
|
|
|
"""Tests an invalid project JSON results in None being returned."""
|
2021-02-03 23:18:56 +00:00
|
|
|
coverage_getter = coverage.OssFuzzCoverageGetter(PROJECT_NAME, REPO_PATH)
|
|
|
|
self.assertIsNone(coverage_getter.get_target_coverage_report(FUZZ_TARGET))
|
2021-01-20 16:22:24 +00:00
|
|
|
|
|
|
|
|
|
|
|
class GetFilesCoveredByTargetTest(unittest.TestCase):
|
|
|
|
"""Tests get_files_covered_by_target."""
|
|
|
|
|
|
|
|
def setUp(self):
|
2021-01-20 16:46:38 +00:00
|
|
|
with mock.patch('coverage._get_latest_cov_report_info',
|
2021-01-20 19:19:15 +00:00
|
|
|
return_value=PROJECT_COV_INFO):
|
2021-01-20 16:46:38 +00:00
|
|
|
self.coverage_getter = coverage.OssFuzzCoverageGetter(
|
2021-01-20 19:19:15 +00:00
|
|
|
PROJECT_NAME, REPO_PATH)
|
2021-01-20 16:22:24 +00:00
|
|
|
|
|
|
|
def test_valid_target(self):
|
|
|
|
"""Tests that covered files can be retrieved from a coverage report."""
|
2021-03-19 18:49:38 +00:00
|
|
|
with open(os.path.join(TEST_DATA_PATH,
|
2021-01-20 19:24:25 +00:00
|
|
|
FUZZ_TARGET_COV_JSON_FILENAME),) as file_handle:
|
2021-01-20 19:19:15 +00:00
|
|
|
fuzzer_cov_info = json.loads(file_handle.read())
|
|
|
|
|
2021-01-20 16:46:38 +00:00
|
|
|
with mock.patch('coverage.OssFuzzCoverageGetter.get_target_coverage_report',
|
2021-01-20 19:19:15 +00:00
|
|
|
return_value=fuzzer_cov_info):
|
2021-01-20 19:24:25 +00:00
|
|
|
file_list = self.coverage_getter.get_files_covered_by_target(FUZZ_TARGET)
|
2021-01-20 16:22:24 +00:00
|
|
|
|
2021-03-19 18:49:38 +00:00
|
|
|
curl_files_list_path = os.path.join(TEST_DATA_PATH,
|
2021-01-20 16:22:24 +00:00
|
|
|
'example_curl_file_list.json')
|
|
|
|
with open(curl_files_list_path) as file_handle:
|
2021-01-20 20:53:47 +00:00
|
|
|
expected_file_list = json.loads(file_handle.read())
|
2021-01-20 19:19:15 +00:00
|
|
|
self.assertCountEqual(file_list, expected_file_list)
|
|
|
|
|
|
|
|
def test_invalid_target(self):
|
|
|
|
"""Tests passing invalid fuzz target returns None."""
|
|
|
|
self.assertIsNone(
|
2021-01-20 19:24:25 +00:00
|
|
|
self.coverage_getter.get_files_covered_by_target(INVALID_TARGET))
|
2021-01-20 16:22:24 +00:00
|
|
|
|
|
|
|
|
2021-01-20 18:13:42 +00:00
|
|
|
class IsFileCoveredTest(unittest.TestCase):
|
|
|
|
"""Tests for is_file_covered."""
|
2021-01-20 18:16:02 +00:00
|
|
|
|
2021-01-20 18:13:42 +00:00
|
|
|
def test_is_file_covered_covered(self):
|
|
|
|
"""Tests that is_file_covered returns True for a covered file."""
|
|
|
|
file_coverage = {
|
|
|
|
'filename': '/src/systemd/src/basic/locale-util.c',
|
|
|
|
'summary': {
|
|
|
|
'regions': {
|
2021-01-20 18:16:02 +00:00
|
|
|
'count': 204,
|
|
|
|
'covered': 200,
|
|
|
|
'notcovered': 200,
|
2021-01-20 18:13:42 +00:00
|
|
|
'percent': 98.03
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
self.assertTrue(coverage.is_file_covered(file_coverage))
|
|
|
|
|
2021-01-20 18:16:02 +00:00
|
|
|
def test_is_file_covered_not_covered(self):
|
|
|
|
"""Tests that is_file_covered returns False for a not covered file."""
|
2021-01-20 18:13:42 +00:00
|
|
|
file_coverage = {
|
|
|
|
'filename': '/src/systemd/src/basic/locale-util.c',
|
|
|
|
'summary': {
|
|
|
|
'regions': {
|
2021-01-20 18:16:02 +00:00
|
|
|
'count': 204,
|
|
|
|
'covered': 0,
|
|
|
|
'notcovered': 0,
|
2021-01-20 18:13:42 +00:00
|
|
|
'percent': 0
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
self.assertFalse(coverage.is_file_covered(file_coverage))
|
|
|
|
|
|
|
|
|
2021-03-03 22:46:51 +00:00
|
|
|
class GetLatestCovReportInfo(unittest.TestCase):
|
|
|
|
"""Tests that _get_latest_cov_report_info works as intended."""
|
|
|
|
|
|
|
|
PROJECT = 'project'
|
|
|
|
LATEST_REPORT_INFO_URL = ('https://storage.googleapis.com/oss-fuzz-coverage/'
|
|
|
|
'latest_report_info/project.json')
|
|
|
|
|
|
|
|
@mock.patch('logging.error')
|
|
|
|
@mock.patch('coverage.get_json_from_url', return_value={'coverage': 1})
|
|
|
|
def test_get_latest_cov_report_info(self, mocked_get_json_from_url,
|
|
|
|
mocked_error):
|
|
|
|
"""Tests that _get_latest_cov_report_info works as intended."""
|
|
|
|
result = coverage._get_latest_cov_report_info(self.PROJECT)
|
|
|
|
self.assertEqual(result, {'coverage': 1})
|
|
|
|
mocked_error.assert_not_called()
|
|
|
|
mocked_get_json_from_url.assert_called_with(self.LATEST_REPORT_INFO_URL)
|
|
|
|
|
|
|
|
@mock.patch('logging.error')
|
|
|
|
@mock.patch('coverage.get_json_from_url', return_value=None)
|
|
|
|
def test_get_latest_cov_report_info_fail(self, _, mocked_error):
|
|
|
|
"""Tests that _get_latest_cov_report_info works as intended when we can't
|
|
|
|
get latest report info."""
|
|
|
|
result = coverage._get_latest_cov_report_info('project')
|
|
|
|
self.assertIsNone(result)
|
|
|
|
mocked_error.assert_called_with(
|
|
|
|
'Could not get the coverage report json from url: %s.',
|
|
|
|
self.LATEST_REPORT_INFO_URL)
|
|
|
|
|
|
|
|
|
2021-01-20 15:38:05 +00:00
|
|
|
if __name__ == '__main__':
|
|
|
|
unittest.main()
|