diff options
Diffstat (limited to 'api/coverage/tools/ExtractFlaggedApis.kt')
-rw-r--r-- | api/coverage/tools/ExtractFlaggedApis.kt | 101 |
1 files changed, 64 insertions, 37 deletions
diff --git a/api/coverage/tools/ExtractFlaggedApis.kt b/api/coverage/tools/ExtractFlaggedApis.kt index caa1929abd54..5efda98a1518 100644 --- a/api/coverage/tools/ExtractFlaggedApis.kt +++ b/api/coverage/tools/ExtractFlaggedApis.kt @@ -16,51 +16,78 @@ package android.platform.coverage +import com.android.tools.metalava.model.ClassItem +import com.android.tools.metalava.model.Item +import com.android.tools.metalava.model.MethodItem import com.android.tools.metalava.model.text.ApiFile import java.io.File import java.io.FileWriter /** Usage: extract-flagged-apis <api text file> <output .pb file> */ fun main(args: Array<String>) { - var cb = ApiFile.parseApi(listOf(File(args[0]))) - var builder = FlagApiMap.newBuilder() + val cb = ApiFile.parseApi(listOf(File(args[0]))) + val builder = FlagApiMap.newBuilder() for (pkg in cb.getPackages().packages) { - var packageName = pkg.qualifiedName() - pkg.allClasses() - .filter { it.methods().size > 0 } - .forEach { - for (method in it.methods()) { - val flagValue = - method.modifiers - .findAnnotation("android.annotation.FlaggedApi") - ?.findAttribute("value") - ?.value - ?.value() - if (flagValue != null && flagValue is String) { - var api = - JavaMethod.newBuilder() - .setPackageName(packageName) - .setClassName(it.fullName()) - .setMethodName(method.name()) - for (param in method.parameters()) { - api.addParameters(param.type().toTypeString()) - } - if (builder.containsFlagToApi(flagValue)) { - var updatedApis = - builder - .getFlagToApiOrThrow(flagValue) - .toBuilder() - .addJavaMethods(api) - .build() - builder.putFlagToApi(flagValue, updatedApis) - } else { - var apis = FlaggedApis.newBuilder().addJavaMethods(api).build() - builder.putFlagToApi(flagValue, apis) - } - } - } - } + val packageName = pkg.qualifiedName() + pkg.allClasses().forEach { + extractFlaggedApisFromClass(it, it.methods(), packageName, builder) + extractFlaggedApisFromClass(it, it.constructors(), packageName, builder) + } } val flagApiMap = builder.build() FileWriter(args[1]).use { it.write(flagApiMap.toString()) } } + +fun extractFlaggedApisFromClass( + classItem: ClassItem, + methods: List<MethodItem>, + packageName: String, + builder: FlagApiMap.Builder +) { + if (methods.isEmpty()) return + val classFlag = getClassFlag(classItem) + for (method in methods) { + val methodFlag = getFlagAnnotation(method) ?: classFlag + val api = + JavaMethod.newBuilder() + .setPackageName(packageName) + .setClassName(classItem.fullName()) + .setMethodName(method.name()) + for (param in method.parameters()) { + api.addParameters(param.type().toTypeString()) + } + if (methodFlag != null) { + addFlaggedApi(builder, api, methodFlag) + } + } +} + +fun addFlaggedApi(builder: FlagApiMap.Builder, api: JavaMethod.Builder, flag: String) { + if (builder.containsFlagToApi(flag)) { + val updatedApis = builder.getFlagToApiOrThrow(flag).toBuilder().addJavaMethods(api).build() + builder.putFlagToApi(flag, updatedApis) + } else { + val apis = FlaggedApis.newBuilder().addJavaMethods(api).build() + builder.putFlagToApi(flag, apis) + } +} + +fun getClassFlag(classItem: ClassItem): String? { + var classFlag = getFlagAnnotation(classItem) + var cur = classItem + // If a class is not an inner class, use its @FlaggedApi annotation value. + // Otherwise, use the flag value of the closest outer class that is annotated by @FlaggedApi. + while (cur.isInnerClass() && classFlag == null) { + cur = cur.parent() as ClassItem + classFlag = getFlagAnnotation(cur) + } + return classFlag +} + +fun getFlagAnnotation(item: Item): String? { + return item.modifiers + .findAnnotation("android.annotation.FlaggedApi") + ?.findAttribute("value") + ?.value + ?.value() as? String +} |