All Downloads are FREE. Search and download functionalities are using the official Maven repository.

tagan.lang-ksp.1.2.1.source-code.KspAnnotationImpl.kt Maven / Gradle / Ivy

Go to download

Yatagan is a Dependency Injection framework, specializing on runtime performance and build speed. Supports code generation (apt/kapt/ksp) or reflection.

There is a newer version: 1.5.1
Show newest version
/*
 * Copyright 2022 Yandex LLC
 *
 * Licensed under the Apache License, Version 2.0 (the "License");
 * you may not use this file except in compliance with the License.
 * You may obtain a copy of the License at
 *
 * http://www.apache.org/licenses/LICENSE-2.0
 *
 * Unless required by applicable law or agreed to in writing, software
 * distributed under the License is distributed on an "AS IS" BASIS,
 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
 * See the License for the specific language governing permissions and
 * limitations under the License.
 */

package com.yandex.yatagan.lang.ksp

import com.google.devtools.ksp.getClassDeclarationByName
import com.google.devtools.ksp.symbol.ClassKind
import com.google.devtools.ksp.symbol.KSAnnotation
import com.google.devtools.ksp.symbol.KSClassDeclaration
import com.google.devtools.ksp.symbol.KSPropertyDeclaration
import com.google.devtools.ksp.symbol.KSType
import com.yandex.yatagan.base.ObjectCache
import com.yandex.yatagan.base.memoize
import com.yandex.yatagan.lang.Annotation
import com.yandex.yatagan.lang.Annotation.Value
import com.yandex.yatagan.lang.AnnotationDeclaration
import com.yandex.yatagan.lang.Type
import com.yandex.yatagan.lang.compiled.CtAnnotationBase
import com.yandex.yatagan.lang.compiled.CtAnnotationDeclarationBase
import java.lang.annotation.RetentionPolicy
import kotlin.LazyThreadSafetyMode.PUBLICATION

internal class KspAnnotationImpl(
    private val impl: KSAnnotation,
) : CtAnnotationBase() {
    private val descriptor by lazy {
        [email protected]()
    }

    override val annotationClass: AnnotationDeclaration by lazy {
        AnnotationClassImpl(
            declaration = checkNotNull(impl.annotationType.resolve().classDeclaration()),
        )
    }

    override val platformModel: KSAnnotation
        get() = impl

    override fun getValue(attribute: AnnotationDeclaration.Attribute): Value {
        require(attribute is AttributeImpl) { "Invalid attribute type" }
        val arg = impl.arguments.find { (it.name?.asString() ?: "value") == attribute.name }
        requireNotNull(arg) {
            "Attribute with the name ${attribute.name} is missing"
        }
        return ValueImpl(arg.value)
    }

    override fun hashCode(): Int = descriptor.hashCode()

    override fun equals(other: Any?): Boolean {
        return this === other || (other is KspAnnotationImpl && descriptor == other.descriptor)
    }

    private class ValueImpl(
        private val value: Any?,
    ) : ValueBase() {
        private val identity by lazy(PUBLICATION) {
            accept(object : Value.Visitor {
                override fun visitBoolean(value: Boolean) = value
                override fun visitByte(value: Byte) = value
                override fun visitShort(value: Short) = value
                override fun visitInt(value: Int) = value
                override fun visitLong(value: Long) = value
                override fun visitChar(value: Char) = value
                override fun visitFloat(value: Float) = value
                override fun visitDouble(value: Double) = value
                override fun visitString(value: String) = value
                override fun visitType(value: Type) = value
                override fun visitAnnotation(value: Annotation) = value
                override fun visitEnumConstant(enum: Type, constant: String) = enum to constant
                override fun visitArray(value: List) = value
                override fun visitUnresolved() = null
            })
        }

        override val platformModel: Any?
            get() = value

        override fun  accept(visitor: Value.Visitor): R {
            return when(value) {
                is Boolean -> visitor.visitBoolean(value)
                is Byte -> visitor.visitByte(value)
                is Char -> visitor.visitChar(value)
                is Double -> visitor.visitDouble(value)
                is Float -> visitor.visitFloat(value)
                is Int -> visitor.visitInt(value)
                is Long -> visitor.visitLong(value)
                is Short -> visitor.visitShort(value)
                is String -> visitor.visitString(value)
                is KSType -> {
                    val declaration = value.declaration as? KSClassDeclaration
                    if (declaration != null && declaration.classKind == ClassKind.ENUM_ENTRY) {
                        val enumDeclaration = declaration.parentDeclaration as KSClassDeclaration
                        visitor.visitEnumConstant(
                            enum = KspTypeImpl(impl = enumDeclaration.asType(emptyList())),
                            constant = declaration.simpleName.asString(),
                        )
                    } else {
                        visitor.visitType(KspTypeImpl(impl = value))
                    }
                }
                is KSAnnotation -> visitor.visitAnnotation(KspAnnotationImpl(value))
                is List<*> -> visitor.visitArray(value.map { ValueImpl(it ?: "") })
                is Enum<*> -> {
                    // Sometimes KSP yields enums (of platform types?) literally.
                    // Suppress before https://youtrack.jetbrains.com/issue/KT-54005 gets rolled out.
                    @Suppress("ENUM_DECLARING_CLASS_DEPRECATED_WARNING")
                    val enumClass = KspTypeImpl(
                        checkNotNull(Utils.resolver.getClassDeclarationByName(value.declaringClass.canonicalName)) {
                            "enum constant $value has unresolved class (?)"
                        }.asStarProjectedType()
                    )

                    visitor.visitEnumConstant(
                        enum = enumClass,
                        constant = value.name,
                    )
                }
                null -> visitor.visitUnresolved()
                else -> throw AssertionError("Unexpected value type: $value with class ${value.javaClass}")
            }
        }

        override fun equals(other: Any?) = this === other ||
                (other is ValueImpl && other.identity == other.identity)

        override fun hashCode() = identity.hashCode()
    }

    private class AttributeImpl(
        val impl: KSPropertyDeclaration,
    ) : AnnotationDeclaration.Attribute {

        override val name: String
            get() = impl.simpleName.asString()

        override val type: Type by lazy {
            val kotlinType = impl.type.resolve().resolveAliasIfNeeded()
            val declaration = kotlinType.declaration as? KSClassDeclaration
            if (declaration?.qualifiedName?.asString() == "kotlin.reflect.KClass") {
                // Replacing `kotlin.reflect.KClass<*>` with `java.lang.Class`
                KspTypeImpl(
                    impl = Utils.classType.asType(kotlinType.arguments),
                )
            } else {
                KspTypeImpl(
                    reference = impl.type,
                    jvmSignatureHint = Utils.resolver.mapToJvmSignature(impl),
                )
            }
        }
    }

    internal class AnnotationClassImpl private constructor(
        declaration: KSClassDeclaration,
    ) : CtAnnotationDeclarationBase() {
        private val annotated = KspAnnotatedImpl(declaration)

        override val annotations: Sequence
            get() = annotated.annotations

        override fun  isAnnotatedWith(type: Class): Boolean {
            return annotated.isAnnotatedWith(type)
        }

        override val attributes: Sequence by lazy {
            annotated.impl.getAllProperties().map {
                AttributeImpl(impl = it)
            }.memoize()
        }

        override val qualifiedName: String
            get() = annotated.impl.qualifiedName?.asString() ?: ""

        override fun getRetention(): AnnotationRetention {
            for (annotation in annotated.impl.annotations) {
                return when (annotation.annotationType.resolve().resolveAliasIfNeeded().declaration) {
                    Utils.kotlinRetentionClass -> when (val value = annotation["value"]) {
                        is AnnotationRetention -> value
                        is KSType -> when (value.declaration.simpleName.getShortName()) {
                            "SOURCE" -> AnnotationRetention.SOURCE
                            "BINARY" -> AnnotationRetention.BINARY
                            "RUNTIME" -> AnnotationRetention.RUNTIME
                            else -> throw AssertionError("Unexpected retention")
                        }
                        else -> throw AssertionError("Unexpected retention")
                    }
                    Utils.javaRetentionClass -> when (val value = annotation["value"]) {
                        is RetentionPolicy -> when (value) {
                            RetentionPolicy.SOURCE -> AnnotationRetention.SOURCE
                            RetentionPolicy.CLASS -> AnnotationRetention.BINARY
                            RetentionPolicy.RUNTIME -> AnnotationRetention.RUNTIME
                        }
                        is KSType -> when (value.declaration.simpleName.getShortName()) {
                            "SOURCE" -> AnnotationRetention.SOURCE
                            "CLASS" -> AnnotationRetention.BINARY
                            "RUNTIME" -> AnnotationRetention.RUNTIME
                            else -> throw AssertionError("Unexpected retention")
                        }
                        else -> throw AssertionError("Unexpected retention")
                    }
                    else -> continue
                }
            }

            // Defaults:
            return if (annotated.impl.isFromKotlin) {
                // Kotlin default
                AnnotationRetention.RUNTIME
            } else {
                // Java default
                AnnotationRetention.BINARY
            }
        }

        companion object Factory : ObjectCache() {
            operator fun invoke(
                declaration: KSClassDeclaration,
            ): AnnotationClassImpl {
                return createCached(declaration, ::AnnotationClassImpl)
            }
        }
    }
}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy