1*d57664e9SAndroid Build Coastguard Worker /*
2*d57664e9SAndroid Build Coastguard Worker * Copyright (C) 2023 The Android Open Source Project
3*d57664e9SAndroid Build Coastguard Worker *
4*d57664e9SAndroid Build Coastguard Worker * Licensed under the Apache License, Version 2.0 (the "License");
5*d57664e9SAndroid Build Coastguard Worker * you may not use this file except in compliance with the License.
6*d57664e9SAndroid Build Coastguard Worker * You may obtain a copy of the License at
7*d57664e9SAndroid Build Coastguard Worker *
8*d57664e9SAndroid Build Coastguard Worker * http://www.apache.org/licenses/LICENSE-2.0
9*d57664e9SAndroid Build Coastguard Worker *
10*d57664e9SAndroid Build Coastguard Worker * Unless required by applicable law or agreed to in writing, software
11*d57664e9SAndroid Build Coastguard Worker * distributed under the License is distributed on an "AS IS" BASIS,
12*d57664e9SAndroid Build Coastguard Worker * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13*d57664e9SAndroid Build Coastguard Worker * See the License for the specific language governing permissions and
14*d57664e9SAndroid Build Coastguard Worker * limitations under the License.
15*d57664e9SAndroid Build Coastguard Worker */
16*d57664e9SAndroid Build Coastguard Worker
17*d57664e9SAndroid Build Coastguard Worker package android.platform.coverage
18*d57664e9SAndroid Build Coastguard Worker
19*d57664e9SAndroid Build Coastguard Worker import com.android.tools.metalava.model.CallableItem
20*d57664e9SAndroid Build Coastguard Worker import com.android.tools.metalava.model.ClassItem
21*d57664e9SAndroid Build Coastguard Worker import com.android.tools.metalava.model.Item
22*d57664e9SAndroid Build Coastguard Worker import com.android.tools.metalava.model.text.ApiFile
23*d57664e9SAndroid Build Coastguard Worker import java.io.File
24*d57664e9SAndroid Build Coastguard Worker import java.io.FileWriter
25*d57664e9SAndroid Build Coastguard Worker
26*d57664e9SAndroid Build Coastguard Worker /** Usage: extract-flagged-apis <api text file> <output .pb file> */
mainnull27*d57664e9SAndroid Build Coastguard Worker fun main(args: Array<String>) {
28*d57664e9SAndroid Build Coastguard Worker val cb = ApiFile.parseApi(listOf(File(args[0])))
29*d57664e9SAndroid Build Coastguard Worker val builder = FlagApiMap.newBuilder()
30*d57664e9SAndroid Build Coastguard Worker for (pkg in cb.getPackages().packages) {
31*d57664e9SAndroid Build Coastguard Worker val packageName = pkg.qualifiedName()
32*d57664e9SAndroid Build Coastguard Worker pkg.allClasses().forEach {
33*d57664e9SAndroid Build Coastguard Worker extractFlaggedApisFromClass(it, it.methods(), packageName, builder)
34*d57664e9SAndroid Build Coastguard Worker extractFlaggedApisFromClass(it, it.constructors(), packageName, builder)
35*d57664e9SAndroid Build Coastguard Worker }
36*d57664e9SAndroid Build Coastguard Worker }
37*d57664e9SAndroid Build Coastguard Worker val flagApiMap = builder.build()
38*d57664e9SAndroid Build Coastguard Worker FileWriter(args[1]).use { it.write(flagApiMap.toString()) }
39*d57664e9SAndroid Build Coastguard Worker }
40*d57664e9SAndroid Build Coastguard Worker
extractFlaggedApisFromClassnull41*d57664e9SAndroid Build Coastguard Worker fun extractFlaggedApisFromClass(
42*d57664e9SAndroid Build Coastguard Worker classItem: ClassItem,
43*d57664e9SAndroid Build Coastguard Worker callables: List<CallableItem>,
44*d57664e9SAndroid Build Coastguard Worker packageName: String,
45*d57664e9SAndroid Build Coastguard Worker builder: FlagApiMap.Builder
46*d57664e9SAndroid Build Coastguard Worker ) {
47*d57664e9SAndroid Build Coastguard Worker if (callables.isEmpty()) return
48*d57664e9SAndroid Build Coastguard Worker val classFlag = getClassFlag(classItem)
49*d57664e9SAndroid Build Coastguard Worker for (callable in callables) {
50*d57664e9SAndroid Build Coastguard Worker val callableFlag = getFlagAnnotation(callable) ?: classFlag
51*d57664e9SAndroid Build Coastguard Worker val api =
52*d57664e9SAndroid Build Coastguard Worker JavaMethod.newBuilder()
53*d57664e9SAndroid Build Coastguard Worker .setPackageName(packageName)
54*d57664e9SAndroid Build Coastguard Worker .setClassName(classItem.fullName())
55*d57664e9SAndroid Build Coastguard Worker .setMethodName(callable.name())
56*d57664e9SAndroid Build Coastguard Worker for (param in callable.parameters()) {
57*d57664e9SAndroid Build Coastguard Worker api.addParameters(param.type().toTypeString())
58*d57664e9SAndroid Build Coastguard Worker }
59*d57664e9SAndroid Build Coastguard Worker if (callableFlag != null) {
60*d57664e9SAndroid Build Coastguard Worker addFlaggedApi(builder, api, callableFlag)
61*d57664e9SAndroid Build Coastguard Worker }
62*d57664e9SAndroid Build Coastguard Worker }
63*d57664e9SAndroid Build Coastguard Worker }
64*d57664e9SAndroid Build Coastguard Worker
addFlaggedApinull65*d57664e9SAndroid Build Coastguard Worker fun addFlaggedApi(builder: FlagApiMap.Builder, api: JavaMethod.Builder, flag: String) {
66*d57664e9SAndroid Build Coastguard Worker if (builder.containsFlagToApi(flag)) {
67*d57664e9SAndroid Build Coastguard Worker val updatedApis = builder.getFlagToApiOrThrow(flag).toBuilder().addJavaMethods(api).build()
68*d57664e9SAndroid Build Coastguard Worker builder.putFlagToApi(flag, updatedApis)
69*d57664e9SAndroid Build Coastguard Worker } else {
70*d57664e9SAndroid Build Coastguard Worker val apis = FlaggedApis.newBuilder().addJavaMethods(api).build()
71*d57664e9SAndroid Build Coastguard Worker builder.putFlagToApi(flag, apis)
72*d57664e9SAndroid Build Coastguard Worker }
73*d57664e9SAndroid Build Coastguard Worker }
74*d57664e9SAndroid Build Coastguard Worker
getClassFlagnull75*d57664e9SAndroid Build Coastguard Worker fun getClassFlag(classItem: ClassItem): String? {
76*d57664e9SAndroid Build Coastguard Worker var classFlag = getFlagAnnotation(classItem)
77*d57664e9SAndroid Build Coastguard Worker var cur = classItem
78*d57664e9SAndroid Build Coastguard Worker // If a class is not a nested class, use its @FlaggedApi annotation value.
79*d57664e9SAndroid Build Coastguard Worker // Otherwise, use the flag value of the closest outer class that is annotated by @FlaggedApi.
80*d57664e9SAndroid Build Coastguard Worker while (classFlag == null) {
81*d57664e9SAndroid Build Coastguard Worker cur = cur.containingClass() ?: break
82*d57664e9SAndroid Build Coastguard Worker classFlag = getFlagAnnotation(cur)
83*d57664e9SAndroid Build Coastguard Worker }
84*d57664e9SAndroid Build Coastguard Worker return classFlag
85*d57664e9SAndroid Build Coastguard Worker }
86*d57664e9SAndroid Build Coastguard Worker
getFlagAnnotationnull87*d57664e9SAndroid Build Coastguard Worker fun getFlagAnnotation(item: Item): String? {
88*d57664e9SAndroid Build Coastguard Worker return item.modifiers
89*d57664e9SAndroid Build Coastguard Worker .findAnnotation("android.annotation.FlaggedApi")
90*d57664e9SAndroid Build Coastguard Worker ?.findAttribute("value")
91*d57664e9SAndroid Build Coastguard Worker ?.value
92*d57664e9SAndroid Build Coastguard Worker ?.value() as? String
93*d57664e9SAndroid Build Coastguard Worker }
94