| #!/usr/bin/env python |
| # |
| # Copyright (C) 2018 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. |
| """Verify that one set of hidden API flags is a subset of another.""" |
| |
| import argparse |
| import csv |
| import sys |
| from itertools import chain |
| |
| from signature_trie import signature_trie |
| |
| |
| def dict_reader(csv_file): |
| return csv.DictReader( |
| csv_file, delimiter=",", quotechar="|", fieldnames=["signature"]) |
| |
| |
| def read_flag_trie_from_file(file): |
| with open(file, "r", encoding="utf8") as stream: |
| return read_flag_trie_from_stream(stream) |
| |
| |
| def read_flag_trie_from_stream(stream): |
| trie = signature_trie() |
| reader = dict_reader(stream) |
| for row in reader: |
| signature = row["signature"] |
| trie.add(signature, row) |
| return trie |
| |
| |
| def extract_subset_from_monolithic_flags_as_dict_from_file( |
| monolithic_trie, patterns_file): |
| """Extract a subset of flags from the dict of monolithic flags. |
| |
| :param monolithic_trie: the trie containing all the monolithic flags. |
| :param patterns_file: a file containing a list of signature patterns that |
| define the subset. |
| :return: the dict from signature to row. |
| """ |
| with open(patterns_file, "r", encoding="utf8") as stream: |
| return extract_subset_from_monolithic_flags_as_dict_from_stream( |
| monolithic_trie, stream) |
| |
| |
| def extract_subset_from_monolithic_flags_as_dict_from_stream( |
| monolithic_trie, stream): |
| """Extract a subset of flags from the trie of monolithic flags. |
| |
| :param monolithic_trie: the trie containing all the monolithic flags. |
| :param stream: a stream containing a list of signature patterns that define |
| the subset. |
| :return: the dict from signature to row. |
| """ |
| dict_signature_to_row = {} |
| for pattern in stream: |
| pattern = pattern.rstrip() |
| rows = monolithic_trie.get_matching_rows(pattern) |
| for row in rows: |
| signature = row["signature"] |
| dict_signature_to_row[signature] = row |
| return dict_signature_to_row |
| |
| |
| 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_signature_to_row = {} |
| reader = dict_reader(stream) |
| for row in reader: |
| signature = row["signature"] |
| dict_signature_to_row[signature] = row |
| return dict_signature_to_row |
| |
| |
| def read_signature_csv_from_file_as_dict(csv_file): |
| """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 csv_file: the csv file to read |
| :return: the dict from signature to row. |
| """ |
| with open(csv_file, "r", encoding="utf8") as f: |
| return read_signature_csv_from_stream_as_dict(f) |
| |
| |
| def compare_signature_flags(monolithic_flags_dict, modular_flags_dict, |
| implementation_flags): |
| """Compare the signature flags between the two dicts. |
| |
| :param monolithic_flags_dict: the dict containing the subset of the |
| monolithic flags that should be equal to the modular flags. |
| :param modular_flags_dict: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. |
| """ |
| mismatching_signatures = [] |
| # Create a sorted set of all the signatures from both the monolithic and |
| # modular dicts. |
| all_signatures = sorted( |
| set(chain(monolithic_flags_dict.keys(), modular_flags_dict.keys()))) |
| for signature in all_signatures: |
| monolithic_row = monolithic_flags_dict.get(signature, {}) |
| monolithic_flags = monolithic_row.get(None, []) |
| if signature in modular_flags_dict: |
| modular_row = modular_flags_dict.get(signature, {}) |
| modular_flags = modular_row.get(None, []) |
| else: |
| modular_flags = implementation_flags |
| if monolithic_flags != modular_flags: |
| mismatching_signatures.append( |
| (signature, modular_flags, monolithic_flags)) |
| return mismatching_signatures |
| |
| |
| def main(argv): |
| args_parser = argparse.ArgumentParser( |
| description="Verify that sets of hidden API flags are each a subset of " |
| "the monolithic flag file. For each module this uses the provided " |
| "signature patterns to select a subset of the monolithic flags and " |
| "then it compares that subset against the filtered flags provided by " |
| "the module. If the module's filtered flags does not contain flags for " |
| "a signature then it is assumed to have been filtered out because it " |
| "was not part of an API and so is assumed to have the implementation " |
| "flags.") |
| args_parser.add_argument( |
| "--monolithic-flags", help="The monolithic flag file") |
| args_parser.add_argument( |
| "--module-flags", |
| action="append", |
| help="A colon separated pair of paths. The first is a path to a " |
| "filtered set of flags, and the second is a path to a set of " |
| "signature patterns that identify the set of classes belonging to " |
| "a single bootclasspath_fragment module. Specify once for each module " |
| "that needs to be checked.") |
| args_parser.add_argument( |
| "--implementation-flag", |
| action="append", |
| help="A flag in the set of flags that identifies a signature which is " |
| "not part of an API, i.e. is the signature of a private implementation " |
| "member. Specify as many times as necessary to define the " |
| "implementation flag set. If this is not specified then the " |
| "implementation flag set is empty.") |
| args = args_parser.parse_args(argv[1:]) |
| |
| # Read in all the flags into the trie |
| monolithic_flags_path = args.monolithic_flags |
| monolithic_trie = read_flag_trie_from_file(monolithic_flags_path) |
| |
| implementation_flags = args.implementation_flag or [] |
| |
| # For each subset specified on the command line, create dicts for the flags |
| # provided by the subset and the corresponding flags from the complete set |
| # of flags and compare them. |
| failed = False |
| module_pairs = args.module_flags or [] |
| for modular_pair in module_pairs: |
| parts = modular_pair.split(":") |
| modular_flags_path = parts[0] |
| modular_patterns_path = parts[1] |
| modular_flags_dict = read_signature_csv_from_file_as_dict( |
| modular_flags_path) |
| monolithic_flags_subset_dict = \ |
| extract_subset_from_monolithic_flags_as_dict_from_file( |
| monolithic_trie, modular_patterns_path) |
| mismatching_signatures = compare_signature_flags( |
| monolithic_flags_subset_dict, modular_flags_dict, |
| implementation_flags) |
| if mismatching_signatures: |
| failed = True |
| print("ERROR: Hidden API flags are inconsistent:") |
| print("< " + modular_flags_path) |
| print("> " + monolithic_flags_path) |
| for mismatch in mismatching_signatures: |
| 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) |