blob: 0a3ae4f790b0be693aaab53d049d6d34a47f6920 [file] [log] [blame]
Yiming Pana95134c2023-11-15 00:26:25 +00001/*
2 * Copyright (C) 2023 The Android Open Source Project
3 *
4 * Licensed under the Apache License, Version 2.0 (the "License");
5 * you may not use this file except in compliance with the License.
6 * You may obtain a copy of the License at
7 *
8 * http://www.apache.org/licenses/LICENSE-2.0
9 *
10 * Unless required by applicable law or agreed to in writing, software
11 * distributed under the License is distributed on an "AS IS" BASIS,
12 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13 * See the License for the specific language governing permissions and
14 * limitations under the License.
15 */
16
17package android.platform.coverage
18
Paul Duffina858a712024-07-15 22:51:20 +010019import com.android.tools.metalava.model.CallableItem
Yiming Pan46491352024-03-26 08:01:26 +000020import com.android.tools.metalava.model.ClassItem
Yiming Pandd35c2f2024-04-10 23:07:29 +000021import com.android.tools.metalava.model.Item
Yiming Pana95134c2023-11-15 00:26:25 +000022import com.android.tools.metalava.model.text.ApiFile
23import java.io.File
24import java.io.FileWriter
25
26/** Usage: extract-flagged-apis <api text file> <output .pb file> */
27fun main(args: Array<String>) {
Yiming Pan46491352024-03-26 08:01:26 +000028 val cb = ApiFile.parseApi(listOf(File(args[0])))
29 val builder = FlagApiMap.newBuilder()
Yiming Pan090155e2023-12-08 08:29:41 +000030 for (pkg in cb.getPackages().packages) {
Yiming Pan46491352024-03-26 08:01:26 +000031 val packageName = pkg.qualifiedName()
Yiming Pan823c15d2024-04-01 07:30:27 +000032 pkg.allClasses().forEach {
33 extractFlaggedApisFromClass(it, it.methods(), packageName, builder)
34 extractFlaggedApisFromClass(it, it.constructors(), packageName, builder)
35 }
Yiming Pana95134c2023-11-15 00:26:25 +000036 }
37 val flagApiMap = builder.build()
38 FileWriter(args[1]).use { it.write(flagApiMap.toString()) }
39}
Yiming Pan46491352024-03-26 08:01:26 +000040
41fun extractFlaggedApisFromClass(
42 classItem: ClassItem,
Paul Duffina858a712024-07-15 22:51:20 +010043 callables: List<CallableItem>,
Yiming Pan46491352024-03-26 08:01:26 +000044 packageName: String,
45 builder: FlagApiMap.Builder
46) {
Paul Duffina858a712024-07-15 22:51:20 +010047 if (callables.isEmpty()) return
Yiming Pandd35c2f2024-04-10 23:07:29 +000048 val classFlag = getClassFlag(classItem)
Paul Duffina858a712024-07-15 22:51:20 +010049 for (callable in callables) {
50 val callableFlag = getFlagAnnotation(callable) ?: classFlag
Yiming Pan46491352024-03-26 08:01:26 +000051 val api =
52 JavaMethod.newBuilder()
53 .setPackageName(packageName)
54 .setClassName(classItem.fullName())
Paul Duffina858a712024-07-15 22:51:20 +010055 .setMethodName(callable.name())
56 for (param in callable.parameters()) {
Yiming Pan46491352024-03-26 08:01:26 +000057 api.addParameters(param.type().toTypeString())
58 }
Paul Duffina858a712024-07-15 22:51:20 +010059 if (callableFlag != null) {
60 addFlaggedApi(builder, api, callableFlag)
Yiming Pan46491352024-03-26 08:01:26 +000061 }
62 }
63}
64
65fun addFlaggedApi(builder: FlagApiMap.Builder, api: JavaMethod.Builder, flag: String) {
66 if (builder.containsFlagToApi(flag)) {
67 val updatedApis = builder.getFlagToApiOrThrow(flag).toBuilder().addJavaMethods(api).build()
68 builder.putFlagToApi(flag, updatedApis)
69 } else {
70 val apis = FlaggedApis.newBuilder().addJavaMethods(api).build()
71 builder.putFlagToApi(flag, apis)
72 }
73}
Yiming Pandd35c2f2024-04-10 23:07:29 +000074
75fun getClassFlag(classItem: ClassItem): String? {
76 var classFlag = getFlagAnnotation(classItem)
77 var cur = classItem
Paul Duffinecb807b2024-07-02 15:03:00 +010078 // If a class is not a nested class, use its @FlaggedApi annotation value.
Yiming Pandd35c2f2024-04-10 23:07:29 +000079 // Otherwise, use the flag value of the closest outer class that is annotated by @FlaggedApi.
Paul Duffinecb807b2024-07-02 15:03:00 +010080 while (classFlag == null) {
81 cur = cur.containingClass() ?: break
Yiming Pandd35c2f2024-04-10 23:07:29 +000082 classFlag = getFlagAnnotation(cur)
83 }
84 return classFlag
85}
86
87fun getFlagAnnotation(item: Item): String? {
88 return item.modifiers
89 .findAnnotation("android.annotation.FlaggedApi")
90 ?.findAttribute("value")
91 ?.value
92 ?.value() as? String
93}