com.flyjingfish.android_aop_ksp.AndroidAopSymbolProcessor.kt Maven / Gradle / Ivy
Go to download
Show more of this group Show more artifacts with this name
Show all versions of android-aop-ksp Show documentation
Show all versions of android-aop-ksp Show documentation
Lightweight Aop for Android platform, you deserve it, action is worse than your heartbeat
package com.flyjingfish.android_aop_ksp
import com.flyjingfish.android_aop_annotation.anno.AndroidAopClass
import com.flyjingfish.android_aop_annotation.anno.AndroidAopMatch
import com.flyjingfish.android_aop_annotation.anno.AndroidAopMatchClassMethod
import com.flyjingfish.android_aop_annotation.anno.AndroidAopMethod
import com.flyjingfish.android_aop_annotation.anno.AndroidAopPointCut
import com.google.devtools.ksp.processing.CodeGenerator
import com.google.devtools.ksp.processing.Dependencies
import com.google.devtools.ksp.processing.KSPLogger
import com.google.devtools.ksp.processing.Resolver
import com.google.devtools.ksp.processing.SymbolProcessor
import com.google.devtools.ksp.symbol.KSAnnotated
import com.google.devtools.ksp.symbol.KSClassDeclaration
import com.google.devtools.ksp.symbol.KSType
import com.google.devtools.ksp.symbol.Origin
import com.google.devtools.ksp.validate
import com.squareup.kotlinpoet.AnnotationSpec
import com.squareup.kotlinpoet.FileSpec
import com.squareup.kotlinpoet.FunSpec
import com.squareup.kotlinpoet.KModifier
import com.squareup.kotlinpoet.TypeSpec
import java.lang.annotation.ElementType
class AndroidAopSymbolProcessor(private val codeGenerator: CodeGenerator,
private val logger: KSPLogger
) : SymbolProcessor {
companion object{
const val packageName = "com.flyjingfish.android_aop_core.aop"
}
override fun process(resolver: Resolver): List {
// logger.error("---------AndroidAopSymbolProcessor---------")
val ret1 = processPointCut(resolver)
val ret2 = processMatch(resolver)
val ret = arrayListOf()
ret.addAll(ret1)
ret.addAll(ret2)
return ret
}
private fun processPointCut(resolver: Resolver): List {
val symbols = resolver.getSymbolsWithAnnotation(AndroidAopPointCut::class.qualifiedName!!)
for (symbol in symbols) {
val annotationMap = getAnnotation(symbol)
val classMethodMap: MutableMap =
annotationMap["@AndroidAopPointCut"] ?: continue
val value: KSType? =
if (classMethodMap["value"] != null) classMethodMap["value"] as KSType else null
val targetClassName: String =
(if (value != null) value.declaration.packageName.asString() + "." + value.toString() else null)
?: continue
val targetMap: MutableMap? = annotationMap["@Target"]
if (targetMap != null) {
val value = targetMap["value"]
if (value is ElementType) {
if (ElementType.METHOD != value) {
throw IllegalArgumentException("注意:请给 $symbol 设置 @Target 为 METHOD ")
}
} else if (value is ArrayList<*>) {
for (s in value) {
if (symbol.origin == Origin.JAVA) {
if ("METHOD" != s.toString()) {
if (value.size > 1) {
throw IllegalArgumentException("注意: $symbol 只可以设置 @Target 为 METHOD 这一种")
} else {
throw IllegalArgumentException("注意:请给 $symbol 设置 @Target 为 METHOD ")
}
}
} else if (symbol.origin == Origin.KOTLIN) {
if ("kotlin.annotation.AnnotationTarget.FUNCTION" != s.toString()) {
throw IllegalArgumentException("注意:请给 $symbol 设置 @Target 为 FUNCTION ")
}
}
}
}
val allowedTargets = targetMap["allowedTargets"]
if (allowedTargets is ArrayList<*>) {
val value: ArrayList<*> = allowedTargets
for (s in value) {
if (symbol.origin == Origin.JAVA) {
if ("METHOD" != s.toString()) {
throw IllegalArgumentException("注意:请给 $symbol 设置 @Target 为 METHOD ")
}
} else if (symbol.origin == Origin.KOTLIN) {
if ("kotlin.annotation.AnnotationTarget.FUNCTION" != s.toString() && "kotlin.annotation.AnnotationTarget.PROPERTY_GETTER" != s.toString() && "kotlin.annotation.AnnotationTarget.PROPERTY_SETTER" != s.toString()) {
throw IllegalArgumentException("注意:$symbol 只可设置 @Target 为 FUNCTION、PROPERTY_SETTER 或 PROPERTY_GETTER")
}
}
}
}
} else {
if (symbol.origin == Origin.JAVA) {
throw IllegalArgumentException("注意:请给 $symbol 设置 @Retention 为 METHOD ")
} else if (symbol.origin == Origin.KOTLIN) {
throw IllegalArgumentException("注意:请给 $symbol 设置 @Retention 为 FUNCTION、PROPERTY_SETTER 或 PROPERTY_GETTER 至少一种")
}
}
val retentionMap: MutableMap? = annotationMap["@Retention"]
if (retentionMap != null) {
val value = retentionMap["value"]
val retention = value.toString()
if ((symbol.origin == Origin.JAVA && "RUNTIME" != retention) || (symbol.origin == Origin.KOTLIN && "kotlin.annotation.AnnotationRetention.RUNTIME" != retention)) {
throw IllegalArgumentException("注意:请给 $symbol 设置 @Retention 为 RUNTIME ")
}
} else {
throw IllegalArgumentException("注意:请给 $symbol 设置 @Retention 为 RUNTIME ")
}
val className = (symbol as KSClassDeclaration).packageName.asString() + "." + symbol
val fileName = "${symbol}\$\$AndroidAopClass";
val typeBuilder = TypeSpec.classBuilder(
fileName
).addModifiers(KModifier.FINAL)
.addAnnotation(AndroidAopClass::class)
val whatsMyName1 = whatsMyName("withinAnnotatedClass")
.addAnnotation(
AnnotationSpec.builder(AndroidAopMethod::class)
.addMember(
"value = %S",
"@$className"
)
.addMember(
"pointCutClassName = %S",
targetClassName
)
.build()
)
typeBuilder.addFunction(whatsMyName1.build())
writeToFile(typeBuilder,fileName)
}
return symbols.filter { !it.validate() }.toList()
}
private fun processMatch(resolver: Resolver): List {
val symbols =
resolver.getSymbolsWithAnnotation(AndroidAopMatchClassMethod::class.qualifiedName!!)
for (symbol in symbols) {
var isMatchClassMethod = false
if (symbol is KSClassDeclaration) {
val typeList = symbol.superTypes.toList()
for (ksTypeReference in typeList) {
if (ksTypeReference.toString() == "MatchClassMethod") {
isMatchClassMethod = true
}
}
}
if (!isMatchClassMethod) {
throw IllegalArgumentException("注意:$symbol 必须实现 MatchClassMethod 接口")
}
val annotationMap = getAnnotation(symbol)
val classMethodMap: MutableMap =
annotationMap["@AndroidAopMatchClassMethod"] ?: continue
val targetClassName: String? = classMethodMap["targetClassName"]?.toString()
val methodNames: ArrayList? =
if (classMethodMap["methodName"] is ArrayList<*>) classMethodMap["methodName"] as ArrayList else null
val excludeClasses: ArrayList? =
if (classMethodMap["excludeClasses"] is ArrayList<*>) classMethodMap["excludeClasses"] as ArrayList else null
val typeStr: String? = classMethodMap["type"]?.toString()
val matchType = typeStr?.substring(typeStr.lastIndexOf(".") + 1) ?: "EXTENDS"
val className = (symbol as KSClassDeclaration).packageName.asString() + "." + symbol
if (targetClassName == null || methodNames == null) {
continue
}
val fileName = "${symbol}\$\$AndroidAopClass";
val typeBuilder = TypeSpec.classBuilder(
fileName
).addModifiers(KModifier.FINAL)
.addAnnotation(AndroidAopClass::class)
val methodNamesBuilder = StringBuilder()
for (i in methodNames.indices) {
methodNamesBuilder.append(methodNames[i])
if (i != methodNames.size - 1) {
methodNamesBuilder.append("-")
}
}
val excludeClassesBuilder = StringBuilder()
if (excludeClasses != null) {
for (i in excludeClasses.indices) {
excludeClassesBuilder.append(excludeClasses[i])
if (i != excludeClasses.size - 1) {
excludeClassesBuilder.append("-")
}
}
}
val whatsMyName1 = whatsMyName("withinAnnotatedClass")
.addAnnotation(
AnnotationSpec.builder(AndroidAopMatch::class)
.addMember(
"baseClassName = %S",
targetClassName
)
.addMember(
"methodNames = %S",
methodNamesBuilder
)
.addMember(
"pointCutClassName = %S",
className
)
.addMember(
"matchType = %S",
matchType
)
.addMember(
"excludeClasses = %S",
excludeClassesBuilder
)
.build()
)
typeBuilder.addFunction(whatsMyName1.build())
writeToFile(typeBuilder,fileName)
}
return symbols.filter { !it.validate() }.toList()
}
private fun writeToFile(typeBuilder: TypeSpec.Builder,fileName:String){
val typeSpec = typeBuilder.build()
val kotlinFile = FileSpec.builder(packageName, fileName).addType(typeSpec)
.build()
codeGenerator
.createNewFile(
Dependencies.ALL_FILES,
packageName,
fileName
)
.writer()
.use { kotlinFile.writeTo(it) }
}
private fun getAnnotation(symbol : KSAnnotated):MutableMap?>{
val map = mutableMapOf?>()
for (annotation in symbol.annotations) {
val annotationName = annotation.toString()
var innerMap = map[annotationName]
if (innerMap == null){
innerMap = mutableMapOf()
map[annotationName] = innerMap
}
for (argument in annotation.arguments) {
innerMap[argument.name?.getShortName().toString()] = argument.value as Any
}
}
return map
}
private fun whatsMyName(name: String): FunSpec.Builder {
return FunSpec.builder(name).addModifiers(KModifier.FINAL)
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy