1*333d2b36SAndroid Build Coastguard Worker#!/usr/bin/env python 2*333d2b36SAndroid Build Coastguard Worker# 3*333d2b36SAndroid Build Coastguard Worker# Copyright (C) 2021 The Android Open Source Project 4*333d2b36SAndroid Build Coastguard Worker# 5*333d2b36SAndroid Build Coastguard Worker# Licensed under the Apache License, Version 2.0 (the "License"); 6*333d2b36SAndroid Build Coastguard Worker# you may not use this file except in compliance with the License. 7*333d2b36SAndroid Build Coastguard Worker# You may obtain a copy of the License at 8*333d2b36SAndroid Build Coastguard Worker# 9*333d2b36SAndroid Build Coastguard Worker# http://www.apache.org/licenses/LICENSE-2.0 10*333d2b36SAndroid Build Coastguard Worker# 11*333d2b36SAndroid Build Coastguard Worker# Unless required by applicable law or agreed to in writing, software 12*333d2b36SAndroid Build Coastguard Worker# distributed under the License is distributed on an "AS IS" BASIS, 13*333d2b36SAndroid Build Coastguard Worker# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. 14*333d2b36SAndroid Build Coastguard Worker# See the License for the specific language governing permissions and 15*333d2b36SAndroid Build Coastguard Worker# limitations under the License. 16*333d2b36SAndroid Build Coastguard Worker"""Generate a set of signature patterns for a bootclasspath_fragment. 17*333d2b36SAndroid Build Coastguard Worker 18*333d2b36SAndroid Build Coastguard WorkerThe patterns are generated from the modular flags produced by the 19*333d2b36SAndroid Build Coastguard Workerbootclasspath_fragment and are used to select a subset of the monolithic flags 20*333d2b36SAndroid Build Coastguard Workeragainst which the modular flags can be compared. 21*333d2b36SAndroid Build Coastguard Worker""" 22*333d2b36SAndroid Build Coastguard Worker 23*333d2b36SAndroid Build Coastguard Workerimport argparse 24*333d2b36SAndroid Build Coastguard Workerimport csv 25*333d2b36SAndroid Build Coastguard Workerimport sys 26*333d2b36SAndroid Build Coastguard Worker 27*333d2b36SAndroid Build Coastguard Worker 28*333d2b36SAndroid Build Coastguard Workerdef dict_reader(csv_file): 29*333d2b36SAndroid Build Coastguard Worker return csv.DictReader( 30*333d2b36SAndroid Build Coastguard Worker csv_file, delimiter=',', quotechar='|', fieldnames=['signature']) 31*333d2b36SAndroid Build Coastguard Worker 32*333d2b36SAndroid Build Coastguard Worker 33*333d2b36SAndroid Build Coastguard Workerdef dot_package_to_slash_package(pkg): 34*333d2b36SAndroid Build Coastguard Worker return pkg.replace('.', '/') 35*333d2b36SAndroid Build Coastguard Worker 36*333d2b36SAndroid Build Coastguard Worker 37*333d2b36SAndroid Build Coastguard Workerdef dot_packages_to_slash_packages(pkgs): 38*333d2b36SAndroid Build Coastguard Worker return [dot_package_to_slash_package(p) for p in pkgs] 39*333d2b36SAndroid Build Coastguard Worker 40*333d2b36SAndroid Build Coastguard Worker 41*333d2b36SAndroid Build Coastguard Workerdef slash_package_to_dot_package(pkg): 42*333d2b36SAndroid Build Coastguard Worker return pkg.replace('/', '.') 43*333d2b36SAndroid Build Coastguard Worker 44*333d2b36SAndroid Build Coastguard Worker 45*333d2b36SAndroid Build Coastguard Workerdef slash_packages_to_dot_packages(pkgs): 46*333d2b36SAndroid Build Coastguard Worker return [slash_package_to_dot_package(p) for p in pkgs] 47*333d2b36SAndroid Build Coastguard Worker 48*333d2b36SAndroid Build Coastguard Worker 49*333d2b36SAndroid Build Coastguard Workerdef is_split_package(split_packages, pkg): 50*333d2b36SAndroid Build Coastguard Worker return split_packages and (pkg in split_packages or '*' in split_packages) 51*333d2b36SAndroid Build Coastguard Worker 52*333d2b36SAndroid Build Coastguard Worker 53*333d2b36SAndroid Build Coastguard Workerdef matched_by_package_prefix_pattern(package_prefixes, prefix): 54*333d2b36SAndroid Build Coastguard Worker for packagePrefix in package_prefixes: 55*333d2b36SAndroid Build Coastguard Worker if prefix == packagePrefix: 56*333d2b36SAndroid Build Coastguard Worker return packagePrefix 57*333d2b36SAndroid Build Coastguard Worker if (prefix.startswith(packagePrefix) and 58*333d2b36SAndroid Build Coastguard Worker prefix[len(packagePrefix)] == '/'): 59*333d2b36SAndroid Build Coastguard Worker return packagePrefix 60*333d2b36SAndroid Build Coastguard Worker return False 61*333d2b36SAndroid Build Coastguard Worker 62*333d2b36SAndroid Build Coastguard Worker 63*333d2b36SAndroid Build Coastguard Workerdef validate_package_is_not_matched_by_package_prefix(package_type, pkg, 64*333d2b36SAndroid Build Coastguard Worker package_prefixes): 65*333d2b36SAndroid Build Coastguard Worker package_prefix = matched_by_package_prefix_pattern(package_prefixes, pkg) 66*333d2b36SAndroid Build Coastguard Worker if package_prefix: 67*333d2b36SAndroid Build Coastguard Worker # A package prefix matches the package. 68*333d2b36SAndroid Build Coastguard Worker package_for_output = slash_package_to_dot_package(pkg) 69*333d2b36SAndroid Build Coastguard Worker package_prefix_for_output = slash_package_to_dot_package(package_prefix) 70*333d2b36SAndroid Build Coastguard Worker return [ 71*333d2b36SAndroid Build Coastguard Worker f'{package_type} {package_for_output} is matched by ' 72*333d2b36SAndroid Build Coastguard Worker f'package prefix {package_prefix_for_output}' 73*333d2b36SAndroid Build Coastguard Worker ] 74*333d2b36SAndroid Build Coastguard Worker return [] 75*333d2b36SAndroid Build Coastguard Worker 76*333d2b36SAndroid Build Coastguard Worker 77*333d2b36SAndroid Build Coastguard Workerdef validate_package_prefixes(split_packages, single_packages, 78*333d2b36SAndroid Build Coastguard Worker package_prefixes): 79*333d2b36SAndroid Build Coastguard Worker # If there are no package prefixes then there is no possible conflict 80*333d2b36SAndroid Build Coastguard Worker # between them and the split packages. 81*333d2b36SAndroid Build Coastguard Worker if len(package_prefixes) == 0: 82*333d2b36SAndroid Build Coastguard Worker return [] 83*333d2b36SAndroid Build Coastguard Worker 84*333d2b36SAndroid Build Coastguard Worker # Check to make sure that the split packages and package prefixes do not 85*333d2b36SAndroid Build Coastguard Worker # overlap. 86*333d2b36SAndroid Build Coastguard Worker errors = [] 87*333d2b36SAndroid Build Coastguard Worker for split_package in split_packages: 88*333d2b36SAndroid Build Coastguard Worker if split_package == '*': 89*333d2b36SAndroid Build Coastguard Worker # A package prefix matches a split package. 90*333d2b36SAndroid Build Coastguard Worker package_prefixes_for_output = ', '.join( 91*333d2b36SAndroid Build Coastguard Worker slash_packages_to_dot_packages(package_prefixes)) 92*333d2b36SAndroid Build Coastguard Worker errors.append( 93*333d2b36SAndroid Build Coastguard Worker "split package '*' conflicts with all package prefixes " 94*333d2b36SAndroid Build Coastguard Worker f'{package_prefixes_for_output}\n' 95*333d2b36SAndroid Build Coastguard Worker ' add split_packages:[] to fix') 96*333d2b36SAndroid Build Coastguard Worker else: 97*333d2b36SAndroid Build Coastguard Worker errs = validate_package_is_not_matched_by_package_prefix( 98*333d2b36SAndroid Build Coastguard Worker 'split package', split_package, package_prefixes) 99*333d2b36SAndroid Build Coastguard Worker errors.extend(errs) 100*333d2b36SAndroid Build Coastguard Worker 101*333d2b36SAndroid Build Coastguard Worker # Check to make sure that the single packages and package prefixes do not 102*333d2b36SAndroid Build Coastguard Worker # overlap. 103*333d2b36SAndroid Build Coastguard Worker for single_package in single_packages: 104*333d2b36SAndroid Build Coastguard Worker errs = validate_package_is_not_matched_by_package_prefix( 105*333d2b36SAndroid Build Coastguard Worker 'single package', single_package, package_prefixes) 106*333d2b36SAndroid Build Coastguard Worker errors.extend(errs) 107*333d2b36SAndroid Build Coastguard Worker return errors 108*333d2b36SAndroid Build Coastguard Worker 109*333d2b36SAndroid Build Coastguard Worker 110*333d2b36SAndroid Build Coastguard Workerdef validate_split_packages(split_packages): 111*333d2b36SAndroid Build Coastguard Worker errors = [] 112*333d2b36SAndroid Build Coastguard Worker if '*' in split_packages and len(split_packages) > 1: 113*333d2b36SAndroid Build Coastguard Worker errors.append('split packages are invalid as they contain both the' 114*333d2b36SAndroid Build Coastguard Worker ' wildcard (*) and specific packages, use the wildcard or' 115*333d2b36SAndroid Build Coastguard Worker ' specific packages, not a mixture') 116*333d2b36SAndroid Build Coastguard Worker return errors 117*333d2b36SAndroid Build Coastguard Worker 118*333d2b36SAndroid Build Coastguard Worker 119*333d2b36SAndroid Build Coastguard Workerdef validate_single_packages(split_packages, single_packages): 120*333d2b36SAndroid Build Coastguard Worker overlaps = [] 121*333d2b36SAndroid Build Coastguard Worker for single_package in single_packages: 122*333d2b36SAndroid Build Coastguard Worker if single_package in split_packages: 123*333d2b36SAndroid Build Coastguard Worker overlaps.append(single_package) 124*333d2b36SAndroid Build Coastguard Worker if overlaps: 125*333d2b36SAndroid Build Coastguard Worker indented = ''.join([f'\n {o}' for o in overlaps]) 126*333d2b36SAndroid Build Coastguard Worker return [ 127*333d2b36SAndroid Build Coastguard Worker f'single_packages and split_packages overlap, please ensure the ' 128*333d2b36SAndroid Build Coastguard Worker f'following packages are only present in one:{indented}' 129*333d2b36SAndroid Build Coastguard Worker ] 130*333d2b36SAndroid Build Coastguard Worker return [] 131*333d2b36SAndroid Build Coastguard Worker 132*333d2b36SAndroid Build Coastguard Worker 133*333d2b36SAndroid Build Coastguard Workerdef produce_patterns_from_file(file, 134*333d2b36SAndroid Build Coastguard Worker split_packages=None, 135*333d2b36SAndroid Build Coastguard Worker single_packages=None, 136*333d2b36SAndroid Build Coastguard Worker package_prefixes=None): 137*333d2b36SAndroid Build Coastguard Worker with open(file, 'r', encoding='utf8') as f: 138*333d2b36SAndroid Build Coastguard Worker return produce_patterns_from_stream(f, split_packages, single_packages, 139*333d2b36SAndroid Build Coastguard Worker package_prefixes) 140*333d2b36SAndroid Build Coastguard Worker 141*333d2b36SAndroid Build Coastguard Worker 142*333d2b36SAndroid Build Coastguard Workerdef produce_patterns_from_stream(stream, 143*333d2b36SAndroid Build Coastguard Worker split_packages=None, 144*333d2b36SAndroid Build Coastguard Worker single_packages=None, 145*333d2b36SAndroid Build Coastguard Worker package_prefixes=None): 146*333d2b36SAndroid Build Coastguard Worker split_packages = set(split_packages or []) 147*333d2b36SAndroid Build Coastguard Worker single_packages = set(single_packages or []) 148*333d2b36SAndroid Build Coastguard Worker package_prefixes = list(package_prefixes or []) 149*333d2b36SAndroid Build Coastguard Worker # Read in all the signatures into a list and remove any unnecessary class 150*333d2b36SAndroid Build Coastguard Worker # and member names. 151*333d2b36SAndroid Build Coastguard Worker patterns = set() 152*333d2b36SAndroid Build Coastguard Worker unmatched_packages = set() 153*333d2b36SAndroid Build Coastguard Worker for row in dict_reader(stream): 154*333d2b36SAndroid Build Coastguard Worker signature = row['signature'] 155*333d2b36SAndroid Build Coastguard Worker text = signature.removeprefix('L') 156*333d2b36SAndroid Build Coastguard Worker # Remove the class specific member signature 157*333d2b36SAndroid Build Coastguard Worker pieces = text.split(';->') 158*333d2b36SAndroid Build Coastguard Worker qualified_class_name = pieces[0] 159*333d2b36SAndroid Build Coastguard Worker pieces = qualified_class_name.rsplit('/', maxsplit=1) 160*333d2b36SAndroid Build Coastguard Worker pkg = pieces[0] 161*333d2b36SAndroid Build Coastguard Worker # If the package is split across multiple modules then it cannot be used 162*333d2b36SAndroid Build Coastguard Worker # to select the subset of the monolithic flags that this module 163*333d2b36SAndroid Build Coastguard Worker # produces. In that case we need to keep the name of the class but can 164*333d2b36SAndroid Build Coastguard Worker # discard any nested class names as an outer class cannot be split 165*333d2b36SAndroid Build Coastguard Worker # across modules. 166*333d2b36SAndroid Build Coastguard Worker # 167*333d2b36SAndroid Build Coastguard Worker # If the package is not split then every class in the package must be 168*333d2b36SAndroid Build Coastguard Worker # provided by this module so there is no need to list the classes 169*333d2b36SAndroid Build Coastguard Worker # explicitly so just use the package name instead. 170*333d2b36SAndroid Build Coastguard Worker if is_split_package(split_packages, pkg): 171*333d2b36SAndroid Build Coastguard Worker # Remove inner class names. 172*333d2b36SAndroid Build Coastguard Worker pieces = qualified_class_name.split('$', maxsplit=1) 173*333d2b36SAndroid Build Coastguard Worker pattern = pieces[0] 174*333d2b36SAndroid Build Coastguard Worker patterns.add(pattern) 175*333d2b36SAndroid Build Coastguard Worker elif pkg in single_packages: 176*333d2b36SAndroid Build Coastguard Worker # Add a * to ensure that the pattern matches the classes in that 177*333d2b36SAndroid Build Coastguard Worker # package. 178*333d2b36SAndroid Build Coastguard Worker pattern = pkg + '/*' 179*333d2b36SAndroid Build Coastguard Worker patterns.add(pattern) 180*333d2b36SAndroid Build Coastguard Worker else: 181*333d2b36SAndroid Build Coastguard Worker unmatched_packages.add(pkg) 182*333d2b36SAndroid Build Coastguard Worker 183*333d2b36SAndroid Build Coastguard Worker # Remove any unmatched packages that would be matched by a package prefix 184*333d2b36SAndroid Build Coastguard Worker # pattern. 185*333d2b36SAndroid Build Coastguard Worker unmatched_packages = [ 186*333d2b36SAndroid Build Coastguard Worker p for p in unmatched_packages 187*333d2b36SAndroid Build Coastguard Worker if not matched_by_package_prefix_pattern(package_prefixes, p) 188*333d2b36SAndroid Build Coastguard Worker ] 189*333d2b36SAndroid Build Coastguard Worker errors = [] 190*333d2b36SAndroid Build Coastguard Worker if unmatched_packages: 191*333d2b36SAndroid Build Coastguard Worker unmatched_packages.sort() 192*333d2b36SAndroid Build Coastguard Worker indented = ''.join([ 193*333d2b36SAndroid Build Coastguard Worker f'\n {slash_package_to_dot_package(p)}' 194*333d2b36SAndroid Build Coastguard Worker for p in unmatched_packages 195*333d2b36SAndroid Build Coastguard Worker ]) 196*333d2b36SAndroid Build Coastguard Worker errors.append('The following packages were unexpected, please add them ' 197*333d2b36SAndroid Build Coastguard Worker 'to one of the hidden_api properties, split_packages, ' 198*333d2b36SAndroid Build Coastguard Worker f'single_packages or package_prefixes:{indented}') 199*333d2b36SAndroid Build Coastguard Worker 200*333d2b36SAndroid Build Coastguard Worker # Remove any patterns that would be matched by a package prefix pattern. 201*333d2b36SAndroid Build Coastguard Worker patterns = [ 202*333d2b36SAndroid Build Coastguard Worker p for p in patterns 203*333d2b36SAndroid Build Coastguard Worker if not matched_by_package_prefix_pattern(package_prefixes, p) 204*333d2b36SAndroid Build Coastguard Worker ] 205*333d2b36SAndroid Build Coastguard Worker # Add the package prefix patterns to the list. Add a ** to ensure that each 206*333d2b36SAndroid Build Coastguard Worker # package prefix pattern will match the classes in that package and all 207*333d2b36SAndroid Build Coastguard Worker # sub-packages. 208*333d2b36SAndroid Build Coastguard Worker patterns = patterns + [f'{p}/**' for p in package_prefixes] 209*333d2b36SAndroid Build Coastguard Worker # Sort the patterns. 210*333d2b36SAndroid Build Coastguard Worker patterns.sort() 211*333d2b36SAndroid Build Coastguard Worker return patterns, errors 212*333d2b36SAndroid Build Coastguard Worker 213*333d2b36SAndroid Build Coastguard Worker 214*333d2b36SAndroid Build Coastguard Workerdef print_and_exit(errors): 215*333d2b36SAndroid Build Coastguard Worker for error in errors: 216*333d2b36SAndroid Build Coastguard Worker print(error) 217*333d2b36SAndroid Build Coastguard Worker sys.exit(1) 218*333d2b36SAndroid Build Coastguard Worker 219*333d2b36SAndroid Build Coastguard Worker 220*333d2b36SAndroid Build Coastguard Workerdef main(args): 221*333d2b36SAndroid Build Coastguard Worker args_parser = argparse.ArgumentParser( 222*333d2b36SAndroid Build Coastguard Worker description='Generate a set of signature patterns ' 223*333d2b36SAndroid Build Coastguard Worker 'that select a subset of monolithic hidden API files.') 224*333d2b36SAndroid Build Coastguard Worker args_parser.add_argument( 225*333d2b36SAndroid Build Coastguard Worker '--flags', 226*333d2b36SAndroid Build Coastguard Worker help='The stub flags file which contains an entry for every dex member', 227*333d2b36SAndroid Build Coastguard Worker ) 228*333d2b36SAndroid Build Coastguard Worker args_parser.add_argument( 229*333d2b36SAndroid Build Coastguard Worker '--split-package', 230*333d2b36SAndroid Build Coastguard Worker action='append', 231*333d2b36SAndroid Build Coastguard Worker help='A package that is split across multiple bootclasspath_fragment ' 232*333d2b36SAndroid Build Coastguard Worker 'modules') 233*333d2b36SAndroid Build Coastguard Worker args_parser.add_argument( 234*333d2b36SAndroid Build Coastguard Worker '--package-prefix', 235*333d2b36SAndroid Build Coastguard Worker action='append', 236*333d2b36SAndroid Build Coastguard Worker help='A package prefix unique to this set of flags') 237*333d2b36SAndroid Build Coastguard Worker args_parser.add_argument( 238*333d2b36SAndroid Build Coastguard Worker '--single-package', 239*333d2b36SAndroid Build Coastguard Worker action='append', 240*333d2b36SAndroid Build Coastguard Worker help='A single package unique to this set of flags') 241*333d2b36SAndroid Build Coastguard Worker args_parser.add_argument('--output', help='Generated signature prefixes') 242*333d2b36SAndroid Build Coastguard Worker args = args_parser.parse_args(args) 243*333d2b36SAndroid Build Coastguard Worker 244*333d2b36SAndroid Build Coastguard Worker split_packages = set( 245*333d2b36SAndroid Build Coastguard Worker dot_packages_to_slash_packages(args.split_package or [])) 246*333d2b36SAndroid Build Coastguard Worker errors = validate_split_packages(split_packages) 247*333d2b36SAndroid Build Coastguard Worker if errors: 248*333d2b36SAndroid Build Coastguard Worker print_and_exit(errors) 249*333d2b36SAndroid Build Coastguard Worker 250*333d2b36SAndroid Build Coastguard Worker single_packages = list( 251*333d2b36SAndroid Build Coastguard Worker dot_packages_to_slash_packages(args.single_package or [])) 252*333d2b36SAndroid Build Coastguard Worker 253*333d2b36SAndroid Build Coastguard Worker errors = validate_single_packages(split_packages, single_packages) 254*333d2b36SAndroid Build Coastguard Worker if errors: 255*333d2b36SAndroid Build Coastguard Worker print_and_exit(errors) 256*333d2b36SAndroid Build Coastguard Worker 257*333d2b36SAndroid Build Coastguard Worker package_prefixes = dot_packages_to_slash_packages(args.package_prefix or []) 258*333d2b36SAndroid Build Coastguard Worker 259*333d2b36SAndroid Build Coastguard Worker errors = validate_package_prefixes(split_packages, single_packages, 260*333d2b36SAndroid Build Coastguard Worker package_prefixes) 261*333d2b36SAndroid Build Coastguard Worker if errors: 262*333d2b36SAndroid Build Coastguard Worker print_and_exit(errors) 263*333d2b36SAndroid Build Coastguard Worker 264*333d2b36SAndroid Build Coastguard Worker patterns = [] 265*333d2b36SAndroid Build Coastguard Worker # Read in all the patterns into a list. 266*333d2b36SAndroid Build Coastguard Worker patterns, errors = produce_patterns_from_file(args.flags, split_packages, 267*333d2b36SAndroid Build Coastguard Worker single_packages, 268*333d2b36SAndroid Build Coastguard Worker package_prefixes) 269*333d2b36SAndroid Build Coastguard Worker 270*333d2b36SAndroid Build Coastguard Worker if errors: 271*333d2b36SAndroid Build Coastguard Worker print_and_exit(errors) 272*333d2b36SAndroid Build Coastguard Worker 273*333d2b36SAndroid Build Coastguard Worker # Write out all the patterns. 274*333d2b36SAndroid Build Coastguard Worker with open(args.output, 'w', encoding='utf8') as outputFile: 275*333d2b36SAndroid Build Coastguard Worker for pattern in patterns: 276*333d2b36SAndroid Build Coastguard Worker outputFile.write(pattern) 277*333d2b36SAndroid Build Coastguard Worker outputFile.write('\n') 278*333d2b36SAndroid Build Coastguard Worker 279*333d2b36SAndroid Build Coastguard Worker 280*333d2b36SAndroid Build Coastguard Workerif __name__ == '__main__': 281*333d2b36SAndroid Build Coastguard Worker main(sys.argv[1:]) 282