Add tests for verify_overlaps script
Refactor verify_overlaps to make it testable and add tests for the comparison. It does make one significant change in behavior which is to read each of the files produced by a bootclasspath_fragment into a dict before comparison, rather than reading and comparing them a row at a time. That allows it to reuse the code to read a CSV into a dict. Bug: 194063708 Test: atest --host verify_overlaps_test m out/soong/hiddenapi/hiddenapi-flags.csv - manually change files to cause difference in flags to check that it detects the differences. Change-Id: Ib70ac87fe089fc25e3bef18f367d4939bfc0cb8d
This commit is contained in:
@@ -20,50 +20,82 @@ Verify that one set of hidden API flags is a subset of another.
|
||||
import argparse
|
||||
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):
|
||||
return csv.DictReader(input, delimiter=',', quotechar='|', fieldnames=['signature'])
|
||||
|
||||
# Read in all the flags into a dict indexed by signature
|
||||
allFlagsBySignature = {}
|
||||
with open(args.all, 'r') as allFlagsFile:
|
||||
allFlagsReader = dict_reader(allFlagsFile)
|
||||
for row in allFlagsReader:
|
||||
def read_signature_csv_from_stream_as_dict(stream):
|
||||
"""
|
||||
Read the csv contents from the stream 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 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']
|
||||
allFlagsBySignature[signature]=row
|
||||
dict[signature] = row
|
||||
return dict
|
||||
|
||||
failed = False
|
||||
for subsetPath in args.subsets:
|
||||
def read_signature_csv_from_file_as_dict(csvFile):
|
||||
"""
|
||||
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 = []
|
||||
with open(subsetPath, 'r') as subsetFlagsFile:
|
||||
subsetReader = dict_reader(subsetFlagsFile)
|
||||
for row in subsetReader:
|
||||
signature = row['signature']
|
||||
if signature in allFlagsBySignature:
|
||||
allFlags = allFlagsBySignature.get(signature)
|
||||
if allFlags != row:
|
||||
mismatchingSignatures.append((signature, row.get(None, []), allFlags.get(None, [])))
|
||||
else:
|
||||
mismatchingSignatures.append((signature, row.get(None, []), []))
|
||||
for signature, modularRow in modularFlagsDict.items():
|
||||
modularFlags = modularRow.get(None, [])
|
||||
monolithicRow = monolithicFlagsDict.get(signature, {})
|
||||
monolithicFlags = monolithicRow.get(None, [])
|
||||
if monolithicFlags != modularFlags:
|
||||
mismatchingSignatures.append((signature, modularFlags, monolithicFlags))
|
||||
return mismatchingSignatures
|
||||
|
||||
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:
|
||||
failed = True
|
||||
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")
|
||||
# Read in all the flags into a dict indexed by signature
|
||||
allFlagsBySignature = read_signature_csv_from_file_as_dict(args.all)
|
||||
|
||||
if failed:
|
||||
sys.exit(1)
|
||||
failed = False
|
||||
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)
|
||||
|
Reference in New Issue
Block a user