xref: /aosp_15_r20/build/soong/scripts/hiddenapi/signature_patterns.py (revision 333d2b3687b3a337dbcca9d65000bca186795e39)
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