| #!/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. | 
 | """Generate a set of signature patterns for a bootclasspath_fragment. | 
 |  | 
 | The patterns are generated from the modular flags produced by the | 
 | bootclasspath_fragment and are used to select a subset of the monolithic flags | 
 | against which the modular flags can be compared. | 
 | """ | 
 |  | 
 | import argparse | 
 | import csv | 
 | import sys | 
 |  | 
 |  | 
 | def dict_reader(csv_file): | 
 |     return csv.DictReader( | 
 |         csv_file, delimiter=',', quotechar='|', fieldnames=['signature']) | 
 |  | 
 |  | 
 | def dot_package_to_slash_package(pkg): | 
 |     return pkg.replace('.', '/') | 
 |  | 
 |  | 
 | def dot_packages_to_slash_packages(pkgs): | 
 |     return [dot_package_to_slash_package(p) for p in pkgs] | 
 |  | 
 |  | 
 | def slash_package_to_dot_package(pkg): | 
 |     return pkg.replace('/', '.') | 
 |  | 
 |  | 
 | def slash_packages_to_dot_packages(pkgs): | 
 |     return [slash_package_to_dot_package(p) for p in pkgs] | 
 |  | 
 |  | 
 | def is_split_package(split_packages, pkg): | 
 |     return split_packages and (pkg in split_packages or '*' in split_packages) | 
 |  | 
 |  | 
 | def matched_by_package_prefix_pattern(package_prefixes, prefix): | 
 |     for packagePrefix in package_prefixes: | 
 |         if prefix == packagePrefix: | 
 |             return packagePrefix | 
 |         if (prefix.startswith(packagePrefix) and | 
 |                 prefix[len(packagePrefix)] == '/'): | 
 |             return packagePrefix | 
 |     return False | 
 |  | 
 |  | 
 | def validate_package_is_not_matched_by_package_prefix(package_type, pkg, | 
 |                                                       package_prefixes): | 
 |     package_prefix = matched_by_package_prefix_pattern(package_prefixes, pkg) | 
 |     if package_prefix: | 
 |         # A package prefix matches the package. | 
 |         package_for_output = slash_package_to_dot_package(pkg) | 
 |         package_prefix_for_output = slash_package_to_dot_package(package_prefix) | 
 |         return [ | 
 |             f'{package_type} {package_for_output} is matched by ' | 
 |             f'package prefix {package_prefix_for_output}' | 
 |         ] | 
 |     return [] | 
 |  | 
 |  | 
 | def validate_package_prefixes(split_packages, single_packages, | 
 |                               package_prefixes): | 
 |     # If there are no package prefixes then there is no possible conflict | 
 |     # between them and the split packages. | 
 |     if len(package_prefixes) == 0: | 
 |         return [] | 
 |  | 
 |     # Check to make sure that the split packages and package prefixes do not | 
 |     # overlap. | 
 |     errors = [] | 
 |     for split_package in split_packages: | 
 |         if split_package == '*': | 
 |             # A package prefix matches a split package. | 
 |             package_prefixes_for_output = ', '.join( | 
 |                 slash_packages_to_dot_packages(package_prefixes)) | 
 |             errors.append( | 
 |                 "split package '*' conflicts with all package prefixes " | 
 |                 f'{package_prefixes_for_output}\n' | 
 |                 '    add split_packages:[] to fix') | 
 |         else: | 
 |             errs = validate_package_is_not_matched_by_package_prefix( | 
 |                 'split package', split_package, package_prefixes) | 
 |             errors.extend(errs) | 
 |  | 
 |     # Check to make sure that the single packages and package prefixes do not | 
 |     # overlap. | 
 |     for single_package in single_packages: | 
 |         errs = validate_package_is_not_matched_by_package_prefix( | 
 |             'single package', single_package, package_prefixes) | 
 |         errors.extend(errs) | 
 |     return errors | 
 |  | 
 |  | 
 | def validate_split_packages(split_packages): | 
 |     errors = [] | 
 |     if '*' in split_packages and len(split_packages) > 1: | 
 |         errors.append('split packages are invalid as they contain both the' | 
 |                       ' wildcard (*) and specific packages, use the wildcard or' | 
 |                       ' specific packages, not a mixture') | 
 |     return errors | 
 |  | 
 |  | 
 | def validate_single_packages(split_packages, single_packages): | 
 |     overlaps = [] | 
 |     for single_package in single_packages: | 
 |         if single_package in split_packages: | 
 |             overlaps.append(single_package) | 
 |     if overlaps: | 
 |         indented = ''.join([f'\n    {o}' for o in overlaps]) | 
 |         return [ | 
 |             f'single_packages and split_packages overlap, please ensure the ' | 
 |             f'following packages are only present in one:{indented}' | 
 |         ] | 
 |     return [] | 
 |  | 
 |  | 
 | def produce_patterns_from_file(file, | 
 |                                split_packages=None, | 
 |                                single_packages=None, | 
 |                                package_prefixes=None): | 
 |     with open(file, 'r', encoding='utf8') as f: | 
 |         return produce_patterns_from_stream(f, split_packages, single_packages, | 
 |                                             package_prefixes) | 
 |  | 
 |  | 
 | def produce_patterns_from_stream(stream, | 
 |                                  split_packages=None, | 
 |                                  single_packages=None, | 
 |                                  package_prefixes=None): | 
 |     split_packages = set(split_packages or []) | 
 |     single_packages = set(single_packages or []) | 
 |     package_prefixes = list(package_prefixes or []) | 
 |     # Read in all the signatures into a list and remove any unnecessary class | 
 |     # and member names. | 
 |     patterns = set() | 
 |     unmatched_packages = set() | 
 |     for row in dict_reader(stream): | 
 |         signature = row['signature'] | 
 |         text = signature.removeprefix('L') | 
 |         # Remove the class specific member signature | 
 |         pieces = text.split(';->') | 
 |         qualified_class_name = pieces[0] | 
 |         pieces = qualified_class_name.rsplit('/', maxsplit=1) | 
 |         pkg = pieces[0] | 
 |         # If the package is split across multiple modules then it cannot be used | 
 |         # to select the subset of the monolithic flags that this module | 
 |         # produces. In that case we need to keep the name of the class but can | 
 |         # discard any nested class names as an outer class cannot be split | 
 |         # across modules. | 
 |         # | 
 |         # If the package is not split then every class in the package must be | 
 |         # provided by this module so there is no need to list the classes | 
 |         # explicitly so just use the package name instead. | 
 |         if is_split_package(split_packages, pkg): | 
 |             # Remove inner class names. | 
 |             pieces = qualified_class_name.split('$', maxsplit=1) | 
 |             pattern = pieces[0] | 
 |             patterns.add(pattern) | 
 |         elif pkg in single_packages: | 
 |             # Add a * to ensure that the pattern matches the classes in that | 
 |             # package. | 
 |             pattern = pkg + '/*' | 
 |             patterns.add(pattern) | 
 |         else: | 
 |             unmatched_packages.add(pkg) | 
 |  | 
 |     # Remove any unmatched packages that would be matched by a package prefix | 
 |     # pattern. | 
 |     unmatched_packages = [ | 
 |         p for p in unmatched_packages | 
 |         if not matched_by_package_prefix_pattern(package_prefixes, p) | 
 |     ] | 
 |     errors = [] | 
 |     if unmatched_packages: | 
 |         unmatched_packages.sort() | 
 |         indented = ''.join([ | 
 |             f'\n    {slash_package_to_dot_package(p)}' | 
 |             for p in unmatched_packages | 
 |         ]) | 
 |         errors.append('The following packages were unexpected, please add them ' | 
 |                       'to one of the hidden_api properties, split_packages, ' | 
 |                       f'single_packages or package_prefixes:{indented}') | 
 |  | 
 |     # Remove any patterns that would be matched by a package prefix pattern. | 
 |     patterns = [ | 
 |         p for p in patterns | 
 |         if not matched_by_package_prefix_pattern(package_prefixes, p) | 
 |     ] | 
 |     # Add the package prefix patterns to the list. Add a ** to ensure that each | 
 |     # package prefix pattern will match the classes in that package and all | 
 |     # sub-packages. | 
 |     patterns = patterns + [f'{p}/**' for p in package_prefixes] | 
 |     # Sort the patterns. | 
 |     patterns.sort() | 
 |     return patterns, errors | 
 |  | 
 |  | 
 | def print_and_exit(errors): | 
 |     for error in errors: | 
 |         print(error) | 
 |     sys.exit(1) | 
 |  | 
 |  | 
 | def main(args): | 
 |     args_parser = argparse.ArgumentParser( | 
 |         description='Generate a set of signature patterns ' | 
 |         'that select a subset of monolithic hidden API files.') | 
 |     args_parser.add_argument( | 
 |         '--flags', | 
 |         help='The stub flags file which contains an entry for every dex member', | 
 |     ) | 
 |     args_parser.add_argument( | 
 |         '--split-package', | 
 |         action='append', | 
 |         help='A package that is split across multiple bootclasspath_fragment ' | 
 |         'modules') | 
 |     args_parser.add_argument( | 
 |         '--package-prefix', | 
 |         action='append', | 
 |         help='A package prefix unique to this set of flags') | 
 |     args_parser.add_argument( | 
 |         '--single-package', | 
 |         action='append', | 
 |         help='A single package unique to this set of flags') | 
 |     args_parser.add_argument('--output', help='Generated signature prefixes') | 
 |     args = args_parser.parse_args(args) | 
 |  | 
 |     split_packages = set( | 
 |         dot_packages_to_slash_packages(args.split_package or [])) | 
 |     errors = validate_split_packages(split_packages) | 
 |     if errors: | 
 |         print_and_exit(errors) | 
 |  | 
 |     single_packages = list( | 
 |         dot_packages_to_slash_packages(args.single_package or [])) | 
 |  | 
 |     errors = validate_single_packages(split_packages, single_packages) | 
 |     if errors: | 
 |         print_and_exit(errors) | 
 |  | 
 |     package_prefixes = dot_packages_to_slash_packages(args.package_prefix or []) | 
 |  | 
 |     errors = validate_package_prefixes(split_packages, single_packages, | 
 |                                        package_prefixes) | 
 |     if errors: | 
 |         print_and_exit(errors) | 
 |  | 
 |     patterns = [] | 
 |     # Read in all the patterns into a list. | 
 |     patterns, errors = produce_patterns_from_file(args.flags, split_packages, | 
 |                                                   single_packages, | 
 |                                                   package_prefixes) | 
 |  | 
 |     if errors: | 
 |         print_and_exit(errors) | 
 |  | 
 |     # Write out all the patterns. | 
 |     with open(args.output, 'w', encoding='utf8') as outputFile: | 
 |         for pattern in patterns: | 
 |             outputFile.write(pattern) | 
 |             outputFile.write('\n') | 
 |  | 
 |  | 
 | if __name__ == '__main__': | 
 |     main(sys.argv[1:]) |