org.jetbrains.kotlin.fir.declarations.FirAnnotationUtils.kt Maven / Gradle / Ivy
Go to download
Show more of this group Show more artifacts with this name
Show all versions of kotlin-compiler-embeddable Show documentation
Show all versions of kotlin-compiler-embeddable Show documentation
the Kotlin compiler embeddable
/*
* Copyright 2010-2021 JetBrains s.r.o. and Kotlin Programming Language contributors.
* Use of this source code is governed by the Apache 2.0 license that can be found in the license/LICENSE.txt file.
*/
package org.jetbrains.kotlin.fir.declarations
import org.jetbrains.kotlin.fir.FirAnnotationContainer
import org.jetbrains.kotlin.fir.FirElement
import org.jetbrains.kotlin.fir.FirEvaluatorResult
import org.jetbrains.kotlin.fir.FirSession
import org.jetbrains.kotlin.fir.expressions.*
import org.jetbrains.kotlin.fir.references.FirErrorNamedReference
import org.jetbrains.kotlin.fir.references.FirNamedReference
import org.jetbrains.kotlin.fir.references.FirResolvedNamedReference
import org.jetbrains.kotlin.fir.references.toResolvedEnumEntrySymbol
import org.jetbrains.kotlin.fir.resolve.fullyExpandedType
import org.jetbrains.kotlin.fir.resolve.toSymbol
import org.jetbrains.kotlin.fir.symbols.ConeClassLikeLookupTag
import org.jetbrains.kotlin.fir.symbols.FirBasedSymbol
import org.jetbrains.kotlin.fir.symbols.impl.FirClassLikeSymbol
import org.jetbrains.kotlin.fir.types.*
import org.jetbrains.kotlin.name.ClassId
import org.jetbrains.kotlin.name.FqName
import org.jetbrains.kotlin.name.Name
import org.jetbrains.kotlin.name.StandardClassIds
import org.jetbrains.kotlin.util.PrivateForInline
// --------------------------- annotation -> type/class ---------------------------
fun FirAnnotation.toAnnotationClassLikeType(session: FirSession): ConeClassLikeType? =
// this cast fails when we have generic-typed annotations @T
(annotationTypeRef.coneType as? ConeClassLikeType)?.fullyExpandedType(session)
private fun FirAnnotation.toAnnotationLookupTag(session: FirSession): ConeClassLikeLookupTag? =
toAnnotationClassLikeType(session)?.lookupTag
private fun FirAnnotation.toAnnotationLookupTagSafe(session: FirSession): ConeClassLikeLookupTag? =
annotationTypeRef.coneTypeSafe()?.fullyExpandedType(session)?.lookupTag
fun FirAnnotation.toAnnotationClassId(session: FirSession): ClassId? =
toAnnotationLookupTag(session)?.classId
fun FirAnnotation.toAnnotationClassIdSafe(session: FirSession): ClassId? =
toAnnotationLookupTagSafe(session)?.classId
fun FirAnnotation.toAnnotationClassLikeSymbol(session: FirSession): FirClassLikeSymbol<*>? =
toAnnotationLookupTag(session)?.toSymbol(session)
fun FirAnnotation.toAnnotationClass(session: FirSession): FirRegularClass? =
toAnnotationClassLikeSymbol(session)?.fir as? FirRegularClass
// --------------------------- annotations lookup/filtering ---------------------------
private val sourceName: Name = Name.identifier("SOURCE")
fun List.nonSourceAnnotations(session: FirSession): List =
this.filter { annotation ->
val firAnnotationClass = annotation.toAnnotationClass(session)
firAnnotationClass != null && firAnnotationClass.symbol.resolvedAnnotationsWithClassIds.none { meta ->
meta.toAnnotationClassId(session) == StandardClassIds.Annotations.Retention &&
meta.findArgumentByName(StandardClassIds.Annotations.ParameterNames.retentionValue)
?.extractEnumValueArgumentInfo()?.enumEntryName == sourceName
}
}
fun FirAnnotationContainer.nonSourceAnnotations(session: FirSession): List =
annotations.nonSourceAnnotations(session)
fun FirDeclaration.hasAnnotation(classId: ClassId, session: FirSession): Boolean {
return annotations.hasAnnotation(classId, session)
}
fun FirDeclaration.hasAnnotationSafe(classId: ClassId, session: FirSession): Boolean {
return annotations.hasAnnotationSafe(classId, session)
}
fun FirBasedSymbol<*>.hasAnnotation(classId: ClassId, session: FirSession): Boolean {
return resolvedAnnotationsWithClassIds.hasAnnotation(classId, session)
}
fun FirAnnotationContainer.hasAnnotation(classId: ClassId, session: FirSession): Boolean {
return annotations.hasAnnotation(classId, session)
}
fun List.hasAnnotation(classId: ClassId, session: FirSession): Boolean {
return this.any { it.toAnnotationClassId(session) == classId }
}
fun List.hasAnnotationSafe(classId: ClassId, session: FirSession): Boolean {
return this.any { it.toAnnotationClassIdSafe(session) == classId }
}
fun FirBasedSymbol.getAnnotationByClassId(
classId: ClassId,
session: FirSession
): FirAnnotation? where D : FirAnnotationContainer, D : FirDeclaration {
return fir.getAnnotationByClassId(classId, session)
}
fun FirAnnotationContainer.getAnnotationByClassId(classId: ClassId, session: FirSession): FirAnnotation? {
return annotations.getAnnotationByClassId(classId, session)
}
fun List.getAnnotationByClassId(classId: ClassId, session: FirSession): FirAnnotation? {
return getAnnotationsByClassId(classId, session).firstOrNull()
}
fun FirAnnotationContainer.getAnnotationsByClassId(classId: ClassId, session: FirSession): List =
annotations.getAnnotationsByClassId(classId, session)
fun List.getAnnotationsByClassId(classId: ClassId, session: FirSession): List {
return filter {
it.doesMatchesClassId(classId, session)
}
}
private fun FirAnnotation.doesMatchesClassId(
classId: ClassId,
session: FirSession,
): Boolean = annotationTypeRef.coneTypeSafe()?.fullyExpandedType(session)?.lookupTag?.classId == classId
fun List.filterOutAnnotationsByClassId(classId: ClassId, session: FirSession): List {
return filterNot {
it.doesMatchesClassId(classId, session)
}
}
fun List.getAnnotationByClassIds(classIds: Collection, session: FirSession): FirAnnotation? {
return firstOrNull {
it.annotationTypeRef.coneTypeSafe()?.fullyExpandedType(session)?.lookupTag?.classId in classIds
}
}
// --------------------------- evaluated arguments ---------------------------
fun FirAnnotation.findArgumentByName(name: Name, returnFirstWhenNotFound: Boolean = true): FirExpression? {
argumentMapping.mapping[name]?.let { return it }
if (this !is FirAnnotationCall) return null
// NB: we have to consider both cases, because deserializer does not create argument mapping
for (argument in arguments) {
if (argument is FirNamedArgumentExpression && argument.name == name) {
return argument.expression
}
}
// The condition is required for annotation arguments that are not fully resolved. For example, CompilerRequiredAnnotations.
// When the annotation is resolved, and we did not find an argument with the given name,
// there is no argument and we should return null.
return if (!resolved && returnFirstWhenNotFound) arguments.firstOrNull() else null
}
fun FirAnnotation.getBooleanArgument(name: Name, session: FirSession): Boolean? = getPrimitiveArgumentValue(name, session)
fun FirAnnotation.getStringArgument(name: Name, session: FirSession): String? = getPrimitiveArgumentValue(name, session)
private inline fun FirAnnotation.getPrimitiveArgumentValue(name: Name, session: FirSession): T? {
val argument = findArgumentByName(name) ?: return null
val literal = argument.evaluateAs(session) ?: return null
return literal.value as? T
}
fun FirAnnotation.getStringArrayArgument(name: Name, session: FirSession): List? {
val argument = findArgumentByName(name) ?: return null
val arrayLiteral = argument.evaluateAs(session) ?: return null
return arrayLiteral.arguments.mapNotNull { (it as? FirLiteralExpression)?.value as? String }
}
fun FirAnnotation.getKClassArgument(name: Name, session: FirSession): ConeKotlinType? {
val argument = findArgumentByName(name) ?: return null
val getClassCall = argument.evaluateAs(session) ?: return null
return getClassCall.getTargetType()
}
fun FirGetClassCall.getTargetType(): ConeKotlinType? {
return resolvedType.typeArguments.getOrNull(0)?.type
}
data class EnumValueArgumentInfo(val enumClassId: ClassId?, val enumEntryName: Name)
fun FirExpression.extractEnumValueArgumentInfo(): EnumValueArgumentInfo? {
return when (this) {
is FirPropertyAccessExpression -> {
if (isResolved) {
val entrySymbol = calleeReference.toResolvedEnumEntrySymbol() ?: return null
EnumValueArgumentInfo(entrySymbol.callableId.classId!!, entrySymbol.callableId.callableName)
} else {
val enumEntryName = (calleeReference as? FirNamedReference)?.name ?: return null
EnumValueArgumentInfo(null, enumEntryName)
}
}
is FirEnumEntryDeserializedAccessExpression -> EnumValueArgumentInfo(enumClassId, enumEntryName)
else -> null
}
}
@PrivateForInline
val FirEvaluatorResult.result: FirElement?
get() = (this as? FirEvaluatorResult.Evaluated)?.result
@OptIn(PrivateForInline::class, PrivateConstantEvaluatorAPI::class)
@Suppress("INVISIBLE_MEMBER", "INVISIBLE_REFERENCE")
inline fun FirExpression.evaluateAs(session: FirSession): @kotlin.internal.NoInfer T? {
return FirExpressionEvaluator.evaluateExpression(this, session)?.result as? T
}
// --------------------------- other utilities ---------------------------
fun FirExpression.unwrapVarargValue(): List {
return when (this) {
is FirVarargArgumentsExpression -> when (val first = arguments.firstOrNull()) {
is FirWrappedArgumentExpression -> first.expression.unwrapVarargValue()
else -> arguments
}
is FirArrayLiteral -> arguments
else -> listOf(this)
}
}
val FirAnnotation.resolved: Boolean
get() {
if (annotationTypeRef !is FirResolvedTypeRef) return false
if (this !is FirAnnotationCall) return true
return calleeReference is FirResolvedNamedReference || calleeReference is FirErrorNamedReference
}
private val LOW_PRIORITY_IN_OVERLOAD_RESOLUTION_CLASS_ID: ClassId =
ClassId(FqName("kotlin.internal"), Name.identifier("LowPriorityInOverloadResolution"))
fun hasLowPriorityAnnotation(annotations: List): Boolean = annotations.any {
val lookupTag = it.annotationTypeRef.coneTypeSafe()?.lookupTag ?: return@any false
lookupTag.classId == LOW_PRIORITY_IN_OVERLOAD_RESOLUTION_CLASS_ID
}