/* * Copyright (C) 2023 The Dagger Authors. * * 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 dagger.hilt.processor.internal.kotlin import androidx.room.compiler.processing.XAnnotation import androidx.room.compiler.processing.XFieldElement import androidx.room.compiler.processing.XMethodElement import androidx.room.compiler.processing.XTypeElement import kotlin.Metadata import kotlin.metadata.declaresDefaultValue import kotlin.metadata.KmClass import kotlin.metadata.KmConstructor import kotlin.metadata.KmFunction import kotlin.metadata.KmProperty import kotlin.metadata.KmValueParameter import kotlin.metadata.jvm.KotlinClassMetadata import kotlin.metadata.jvm.fieldSignature import kotlin.metadata.jvm.getterSignature import kotlin.metadata.jvm.signature import kotlin.metadata.jvm.syntheticMethodForAnnotations /** Container classes for kotlin metadata types. */ class ClassMetadata private constructor(private val kmClass: KmClass) { val functionsBySignature = buildList { addAll(kmClass.constructors.map { ConstructorMetadata(it) }) addAll(kmClass.functions.map { MethodMetadata(it) }) }.associateBy { it.signature } val propertiesBySignature = kmClass.properties .filter { it.fieldSignature != null } .map { PropertyMetadata(it) } .associateBy { it.fieldSignature } fun constructors(): List = functionsBySignature.values.filterIsInstance() companion object { /** Parse Kotlin class metadata from a given type element. */ @JvmStatic fun of(typeElement: XTypeElement): ClassMetadata { val metadataAnnotation = checkNotNull(typeElement.getAnnotation(Metadata::class)).value return when (val classMetadata = KotlinClassMetadata.readStrict(metadataAnnotation)) { is KotlinClassMetadata.Class -> ClassMetadata(classMetadata.kmClass) else -> error("Unsupported metadata type: ${classMetadata}") } } } } class ConstructorMetadata(private val kmConstructor: KmConstructor) : FunctionMetadata { override val name = "" override val signature = kmConstructor.signature!!.toString() override val parameters = kmConstructor.valueParameters.map { ParameterMetadata(it) } } class MethodMetadata(private val kmFunction: KmFunction) : FunctionMetadata { override val name = kmFunction.name override val signature = kmFunction.signature!!.toString() override val parameters = kmFunction.valueParameters.map { ParameterMetadata(it) } } interface FunctionMetadata { val name: String val signature: String val parameters: List } class PropertyMetadata(private val kmProperty: KmProperty) { val name = kmProperty.name /** Returns the JVM field descriptor of the backing field of this property. */ val fieldSignature = kmProperty.fieldSignature?.toString() val getterSignature = kmProperty.getterSignature?.toString() /** Returns JVM method descriptor of the synthetic method for property annotations. */ val methodForAnnotationsSignature = kmProperty.syntheticMethodForAnnotations?.toString() } class ParameterMetadata(private val kmValueParameter: KmValueParameter) { val name = kmValueParameter.name fun declaresDefaultValue() = kmValueParameter.declaresDefaultValue }