Merge "Add tests for verify_overlaps script"
This commit is contained in:
commit
a26cf315a8
3 changed files with 228 additions and 38 deletions
|
@ -83,3 +83,24 @@ python_binary_host {
|
||||||
},
|
},
|
||||||
},
|
},
|
||||||
}
|
}
|
||||||
|
|
||||||
|
python_test_host {
|
||||||
|
name: "verify_overlaps_test",
|
||||||
|
main: "verify_overlaps_test.py",
|
||||||
|
srcs: [
|
||||||
|
"verify_overlaps.py",
|
||||||
|
"verify_overlaps_test.py",
|
||||||
|
],
|
||||||
|
version: {
|
||||||
|
py2: {
|
||||||
|
enabled: false,
|
||||||
|
},
|
||||||
|
py3: {
|
||||||
|
enabled: true,
|
||||||
|
embedded_launcher: true,
|
||||||
|
},
|
||||||
|
},
|
||||||
|
test_options: {
|
||||||
|
unit_test: true,
|
||||||
|
},
|
||||||
|
}
|
||||||
|
|
|
@ -20,50 +20,82 @@ Verify that one set of hidden API flags is a subset of another.
|
||||||
import argparse
|
import argparse
|
||||||
import csv
|
import csv
|
||||||
|
|
||||||
args_parser = argparse.ArgumentParser(description='Verify that one set of hidden API flags is a subset of another.')
|
|
||||||
args_parser.add_argument('all', help='All the flags')
|
|
||||||
args_parser.add_argument('subsets', nargs=argparse.REMAINDER, help='Subsets of the flags')
|
|
||||||
args = args_parser.parse_args()
|
|
||||||
|
|
||||||
|
|
||||||
def dict_reader(input):
|
def dict_reader(input):
|
||||||
return csv.DictReader(input, delimiter=',', quotechar='|', fieldnames=['signature'])
|
return csv.DictReader(input, delimiter=',', quotechar='|', fieldnames=['signature'])
|
||||||
|
|
||||||
# Read in all the flags into a dict indexed by signature
|
def read_signature_csv_from_stream_as_dict(stream):
|
||||||
allFlagsBySignature = {}
|
"""
|
||||||
with open(args.all, 'r') as allFlagsFile:
|
Read the csv contents from the stream into a dict. The first column is assumed to be the
|
||||||
allFlagsReader = dict_reader(allFlagsFile)
|
signature and used as the key. The whole row is stored as the value.
|
||||||
for row in allFlagsReader:
|
|
||||||
|
:param stream: the csv contents to read
|
||||||
|
:return: the dict from signature to row.
|
||||||
|
"""
|
||||||
|
dict = {}
|
||||||
|
reader = dict_reader(stream)
|
||||||
|
for row in reader:
|
||||||
signature = row['signature']
|
signature = row['signature']
|
||||||
allFlagsBySignature[signature]=row
|
dict[signature] = row
|
||||||
|
return dict
|
||||||
|
|
||||||
failed = False
|
def read_signature_csv_from_file_as_dict(csvFile):
|
||||||
for subsetPath in args.subsets:
|
"""
|
||||||
|
Read the csvFile into a dict. The first column is assumed to be the
|
||||||
|
signature and used as the key. The whole row is stored as the value.
|
||||||
|
|
||||||
|
:param csvFile: the csv file to read
|
||||||
|
:return: the dict from signature to row.
|
||||||
|
"""
|
||||||
|
with open(csvFile, 'r') as f:
|
||||||
|
return read_signature_csv_from_stream_as_dict(f)
|
||||||
|
|
||||||
|
def compare_signature_flags(monolithicFlagsDict, modularFlagsDict):
|
||||||
|
"""
|
||||||
|
Compare the signature flags between the two dicts.
|
||||||
|
|
||||||
|
:param monolithicFlagsDict: the dict containing the subset of the monolithic
|
||||||
|
flags that should be equal to the modular flags.
|
||||||
|
:param modularFlagsDict:the dict containing the flags produced by a single
|
||||||
|
bootclasspath_fragment module.
|
||||||
|
:return: list of mismatches., each mismatch is a tuple where the first item
|
||||||
|
is the signature, and the second and third items are lists of the flags from
|
||||||
|
modular dict, and monolithic dict respectively.
|
||||||
|
"""
|
||||||
mismatchingSignatures = []
|
mismatchingSignatures = []
|
||||||
with open(subsetPath, 'r') as subsetFlagsFile:
|
for signature, modularRow in modularFlagsDict.items():
|
||||||
subsetReader = dict_reader(subsetFlagsFile)
|
modularFlags = modularRow.get(None, [])
|
||||||
for row in subsetReader:
|
monolithicRow = monolithicFlagsDict.get(signature, {})
|
||||||
signature = row['signature']
|
monolithicFlags = monolithicRow.get(None, [])
|
||||||
if signature in allFlagsBySignature:
|
if monolithicFlags != modularFlags:
|
||||||
allFlags = allFlagsBySignature.get(signature)
|
mismatchingSignatures.append((signature, modularFlags, monolithicFlags))
|
||||||
if allFlags != row:
|
return mismatchingSignatures
|
||||||
mismatchingSignatures.append((signature, row.get(None, []), allFlags.get(None, [])))
|
|
||||||
else:
|
|
||||||
mismatchingSignatures.append((signature, row.get(None, []), []))
|
|
||||||
|
|
||||||
|
def main(argv):
|
||||||
|
args_parser = argparse.ArgumentParser(description='Verify that one set of hidden API flags is a subset of another.')
|
||||||
|
args_parser.add_argument('all', help='All the flags')
|
||||||
|
args_parser.add_argument('subsets', nargs=argparse.REMAINDER, help='Subsets of the flags')
|
||||||
|
args = args_parser.parse_args(argv[1:])
|
||||||
|
|
||||||
if mismatchingSignatures:
|
# Read in all the flags into a dict indexed by signature
|
||||||
failed = True
|
allFlagsBySignature = read_signature_csv_from_file_as_dict(args.all)
|
||||||
print("ERROR: Hidden API flags are inconsistent:")
|
|
||||||
print("< " + subsetPath)
|
|
||||||
print("> " + args.all)
|
|
||||||
for mismatch in mismatchingSignatures:
|
|
||||||
print()
|
|
||||||
print("< " + mismatch[0] + "," + ",".join(mismatch[1]))
|
|
||||||
if mismatch[2] != []:
|
|
||||||
print("> " + mismatch[0] + "," + ",".join(mismatch[2]))
|
|
||||||
else:
|
|
||||||
print("> " + mismatch[0] + " - missing")
|
|
||||||
|
|
||||||
if failed:
|
failed = False
|
||||||
sys.exit(1)
|
for subsetPath in args.subsets:
|
||||||
|
subsetDict = read_signature_csv_from_file_as_dict(subsetPath)
|
||||||
|
mismatchingSignatures = compare_signature_flags(allFlagsBySignature, subsetDict)
|
||||||
|
if mismatchingSignatures:
|
||||||
|
failed = True
|
||||||
|
print("ERROR: Hidden API flags are inconsistent:")
|
||||||
|
print("< " + subsetPath)
|
||||||
|
print("> " + args.all)
|
||||||
|
for mismatch in mismatchingSignatures:
|
||||||
|
signature = mismatch[0]
|
||||||
|
print()
|
||||||
|
print("< " + ",".join([signature]+ mismatch[1]))
|
||||||
|
print("> " + ",".join([signature]+ mismatch[2]))
|
||||||
|
|
||||||
|
if failed:
|
||||||
|
sys.exit(1)
|
||||||
|
|
||||||
|
if __name__ == "__main__":
|
||||||
|
main(sys.argv)
|
||||||
|
|
137
scripts/hiddenapi/verify_overlaps_test.py
Executable file
137
scripts/hiddenapi/verify_overlaps_test.py
Executable file
|
@ -0,0 +1,137 @@
|
||||||
|
#!/usr/bin/env python
|
||||||
|
#
|
||||||
|
# Copyright (C) 2021 The Android Open Source Project
|
||||||
|
#
|
||||||
|
# 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.
|
||||||
|
|
||||||
|
"""Unit tests for verify_overlaps_test.py."""
|
||||||
|
import io
|
||||||
|
import unittest
|
||||||
|
|
||||||
|
from verify_overlaps import *
|
||||||
|
|
||||||
|
class TestDetectOverlaps(unittest.TestCase):
|
||||||
|
|
||||||
|
def read_signature_csv_from_string_as_dict(self, csv):
|
||||||
|
with io.StringIO(csv) as f:
|
||||||
|
return read_signature_csv_from_stream_as_dict(f)
|
||||||
|
|
||||||
|
def test_match(self):
|
||||||
|
monolithic = self.read_signature_csv_from_string_as_dict('''
|
||||||
|
Ljava/lang/Object;->hashCode()I,public-api,system-api,test-api
|
||||||
|
Ljava/lang/Object;->toString()Ljava/lang/String;,blocked
|
||||||
|
''')
|
||||||
|
modular = self.read_signature_csv_from_string_as_dict('''
|
||||||
|
Ljava/lang/Object;->hashCode()I,public-api,system-api,test-api
|
||||||
|
''')
|
||||||
|
mismatches = compare_signature_flags(monolithic, modular)
|
||||||
|
expected = []
|
||||||
|
self.assertEqual(expected, mismatches)
|
||||||
|
|
||||||
|
def test_mismatch_overlapping_flags(self):
|
||||||
|
monolithic = self.read_signature_csv_from_string_as_dict('''
|
||||||
|
Ljava/lang/Object;->toString()Ljava/lang/String;,public-api
|
||||||
|
''')
|
||||||
|
modular = self.read_signature_csv_from_string_as_dict('''
|
||||||
|
Ljava/lang/Object;->toString()Ljava/lang/String;,public-api,system-api,test-api
|
||||||
|
''')
|
||||||
|
mismatches = compare_signature_flags(monolithic, modular)
|
||||||
|
expected = [
|
||||||
|
(
|
||||||
|
'Ljava/lang/Object;->toString()Ljava/lang/String;',
|
||||||
|
['public-api', 'system-api', 'test-api'],
|
||||||
|
['public-api'],
|
||||||
|
),
|
||||||
|
]
|
||||||
|
self.assertEqual(expected, mismatches)
|
||||||
|
|
||||||
|
|
||||||
|
def test_mismatch_monolithic_blocked(self):
|
||||||
|
monolithic = self.read_signature_csv_from_string_as_dict('''
|
||||||
|
Ljava/lang/Object;->hashCode()I,public-api,system-api,test-api
|
||||||
|
Ljava/lang/Object;->toString()Ljava/lang/String;,blocked
|
||||||
|
''')
|
||||||
|
modular = self.read_signature_csv_from_string_as_dict('''
|
||||||
|
Ljava/lang/Object;->toString()Ljava/lang/String;,public-api,system-api,test-api
|
||||||
|
''')
|
||||||
|
mismatches = compare_signature_flags(monolithic, modular)
|
||||||
|
expected = [
|
||||||
|
(
|
||||||
|
'Ljava/lang/Object;->toString()Ljava/lang/String;',
|
||||||
|
['public-api', 'system-api', 'test-api'],
|
||||||
|
['blocked'],
|
||||||
|
),
|
||||||
|
]
|
||||||
|
self.assertEqual(expected, mismatches)
|
||||||
|
|
||||||
|
def test_mismatch_modular_blocked(self):
|
||||||
|
monolithic = self.read_signature_csv_from_string_as_dict('''
|
||||||
|
Ljava/lang/Object;->hashCode()I,public-api,system-api,test-api
|
||||||
|
Ljava/lang/Object;->toString()Ljava/lang/String;,public-api,system-api,test-api
|
||||||
|
''')
|
||||||
|
modular = self.read_signature_csv_from_string_as_dict('''
|
||||||
|
Ljava/lang/Object;->toString()Ljava/lang/String;,blocked
|
||||||
|
''')
|
||||||
|
mismatches = compare_signature_flags(monolithic, modular)
|
||||||
|
expected = [
|
||||||
|
(
|
||||||
|
'Ljava/lang/Object;->toString()Ljava/lang/String;',
|
||||||
|
['blocked'],
|
||||||
|
['public-api', 'system-api', 'test-api'],
|
||||||
|
),
|
||||||
|
]
|
||||||
|
self.assertEqual(expected, mismatches)
|
||||||
|
|
||||||
|
def test_missing_from_monolithic(self):
|
||||||
|
monolithic = self.read_signature_csv_from_string_as_dict('''
|
||||||
|
Ljava/lang/Object;->hashCode()I,public-api,system-api,test-api
|
||||||
|
''')
|
||||||
|
modular = self.read_signature_csv_from_string_as_dict('''
|
||||||
|
Ljava/lang/Object;->toString()Ljava/lang/String;,public-api,system-api,test-api
|
||||||
|
''')
|
||||||
|
mismatches = compare_signature_flags(monolithic, modular)
|
||||||
|
expected = [
|
||||||
|
(
|
||||||
|
'Ljava/lang/Object;->toString()Ljava/lang/String;',
|
||||||
|
['public-api', 'system-api', 'test-api'],
|
||||||
|
[],
|
||||||
|
),
|
||||||
|
]
|
||||||
|
self.assertEqual(expected, mismatches)
|
||||||
|
|
||||||
|
def test_missing_from_modular(self):
|
||||||
|
# The modular dict defines the set of signatures to compare so an entry
|
||||||
|
# in the monolithic dict that does not have a corresponding entry in the
|
||||||
|
# modular dict is ignored.
|
||||||
|
monolithic = self.read_signature_csv_from_string_as_dict('''
|
||||||
|
Ljava/lang/Object;->hashCode()I,public-api,system-api,test-api
|
||||||
|
''')
|
||||||
|
modular = {}
|
||||||
|
mismatches = compare_signature_flags(monolithic, modular)
|
||||||
|
expected = []
|
||||||
|
self.assertEqual(expected, mismatches)
|
||||||
|
|
||||||
|
def test_blocked_missing_from_modular(self):
|
||||||
|
# The modular dict defines the set of signatures to compare so an entry
|
||||||
|
# in the monolithic dict that does not have a corresponding entry in the
|
||||||
|
# modular dict is ignored.
|
||||||
|
monolithic = self.read_signature_csv_from_string_as_dict('''
|
||||||
|
Ljava/lang/Object;->hashCode()I,blocked
|
||||||
|
''')
|
||||||
|
modular = {}
|
||||||
|
mismatches = compare_signature_flags(monolithic, modular)
|
||||||
|
expected = []
|
||||||
|
self.assertEqual(expected, mismatches)
|
||||||
|
|
||||||
|
if __name__ == '__main__':
|
||||||
|
unittest.main(verbosity=2)
|
Loading…
Reference in a new issue