xref: /aosp_15_r20/frameworks/base/api/coverage/tools/ExtractFlaggedApis.kt (revision d57664e9bc4670b3ecf6748a746a57c557b6bc9e)
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