diff --git a/WORKSPACE b/WORKSPACE index 8ba89d24ea4..b670d99a45c 100644 --- a/WORKSPACE +++ b/WORKSPACE @@ -147,7 +147,7 @@ http_archive( load("@io_bazel_rules_kotlin//kotlin:repositories.bzl", "kotlin_repositories", "kotlinc_version") -KOTLIN_VERSION = "1.9.24" +KOTLIN_VERSION = "2.0.21" # Get from https://github.com/JetBrains/kotlin/releases/ KOTLINC_RELEASE_SHA = "eb7b68e01029fa67bc8d060ee54c12018f2c60ddc438cf21db14517229aa693b" @@ -203,7 +203,7 @@ GRPC_VERSION = "1.2.0" INCAP_VERSION = "0.2" -KSP_VERSION = KOTLIN_VERSION + "-1.0.20" +KSP_VERSION = KOTLIN_VERSION + "-1.0.28" MAVEN_VERSION = "3.3.3" @@ -291,7 +291,7 @@ maven_install( "org.jetbrains.kotlin:kotlin-compiler-embeddable:%s" % KOTLIN_VERSION, "org.jetbrains.kotlin:kotlin-daemon-embeddable:%s" % KOTLIN_VERSION, "org.jetbrains.kotlin:kotlin-stdlib:%s" % KOTLIN_VERSION, - "org.jetbrains.kotlin:kotlin-metadata-jvm:2.0.0-Beta5", + "org.jetbrains.kotlin:kotlin-metadata-jvm:%s" % KOTLIN_VERSION, "org.jspecify:jspecify:1.0.0", "org.mockito:mockito-core:2.28.2", "org.pantsbuild:jarjar:1.7.2", diff --git a/java/dagger/hilt/android/plugin/build.gradle b/java/dagger/hilt/android/plugin/build.gradle index 8627290c34b..e40d4ed9e95 100644 --- a/java/dagger/hilt/android/plugin/build.gradle +++ b/java/dagger/hilt/android/plugin/build.gradle @@ -1,8 +1,8 @@ buildscript { ext { - kotlin_version = "1.9.24" + kotlin_version = "2.0.21" agp_version = System.getenv('AGP_VERSION') ?: "8.1.1" - ksp_version = "$kotlin_version-1.0.20" + ksp_version = "$kotlin_version-1.0.28" pluginArtifactId = 'hilt-android-gradle-plugin' pluginId = 'com.google.dagger.hilt.android' } diff --git a/java/dagger/hilt/processor/internal/kotlin/BUILD b/java/dagger/hilt/processor/internal/kotlin/BUILD index 5fe9052c14e..6429bab8e4b 100644 --- a/java/dagger/hilt/processor/internal/kotlin/BUILD +++ b/java/dagger/hilt/processor/internal/kotlin/BUILD @@ -15,13 +15,16 @@ # Description: # Sources related to Kotlin metadata. -load("@rules_java//java:defs.bzl", "java_library") +load("@io_bazel_rules_kotlin//kotlin:jvm.bzl", "kt_jvm_library") package(default_visibility = ["//:src"]) -java_library( +kt_jvm_library( name = "kotlin", - srcs = glob(["*.java"]), + srcs = glob([ + "*.java", + "*.kt", + ]), deps = [ "//:dagger_with_compiler", "//java/dagger/hilt/processor/internal:classnames", diff --git a/java/dagger/hilt/processor/internal/kotlin/ClassMetadata.kt b/java/dagger/hilt/processor/internal/kotlin/ClassMetadata.kt new file mode 100644 index 00000000000..b3091db01bb --- /dev/null +++ b/java/dagger/hilt/processor/internal/kotlin/ClassMetadata.kt @@ -0,0 +1,102 @@ +/* + * 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) }) + }.associate { it.signature to it } + + val propertiesBySignature = + kmClass.properties + .map { PropertyMetadata(it) } + .associate { it.fieldSignature to it } + .filterKeys { it != null } + + fun constructors(): List = + functionsBySignature.values + .filter { it is ConstructorMetadata } + .toList() + + 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 +} + diff --git a/java/dagger/hilt/processor/internal/kotlin/KotlinMetadata.java b/java/dagger/hilt/processor/internal/kotlin/KotlinMetadata.java index 67b2177a089..47edf6e8ce0 100644 --- a/java/dagger/hilt/processor/internal/kotlin/KotlinMetadata.java +++ b/java/dagger/hilt/processor/internal/kotlin/KotlinMetadata.java @@ -17,37 +17,20 @@ package dagger.hilt.processor.internal.kotlin; import static dagger.internal.codegen.extension.DaggerStreams.toImmutableMap; -import static kotlin.metadata.Flag.ValueParameter.DECLARES_DEFAULT_VALUE; -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 com.google.auto.value.AutoValue; import com.google.auto.value.extension.memoized.Memoized; -import com.google.common.collect.ImmutableList; import com.google.common.collect.ImmutableMap; -import com.google.common.collect.ImmutableSet; -import dagger.hilt.processor.internal.ClassNames; import dagger.internal.codegen.extension.DaggerCollectors; import dagger.internal.codegen.xprocessing.XElements; import java.util.HashMap; import java.util.Map; -import java.util.Objects; import java.util.Optional; import java.util.function.Function; import javax.annotation.Nullable; -import kotlin.Metadata; -import kotlin.metadata.Flag; -import kotlin.metadata.KmClass; -import kotlin.metadata.KmConstructor; -import kotlin.metadata.KmFunction; -import kotlin.metadata.KmProperty; -import kotlin.metadata.jvm.JvmExtensionsKt; -import kotlin.metadata.jvm.JvmFieldSignature; -import kotlin.metadata.jvm.JvmMetadataUtil; -import kotlin.metadata.jvm.JvmMethodSignature; -import kotlin.metadata.jvm.KotlinClassMetadata; /** Data class of a TypeElement and its Kotlin metadata. */ @AutoValue @@ -79,8 +62,8 @@ ImmutableMap methodDescriptors() { @Memoized boolean containsConstructorWithDefaultParam() { return classMetadata().constructors().stream() - .flatMap(constructor -> constructor.parameters().stream()) - .anyMatch(parameter -> parameter.flags(DECLARES_DEFAULT_VALUE)); + .flatMap(constructor -> constructor.getParameters().stream()) + .anyMatch(parameter -> parameter.declaresDefaultValue()); } /** Gets the synthetic method for annotations of a given field element. */ @@ -102,8 +85,7 @@ private Optional getAnnotationMethod(XFieldElement fieldEl } private Optional getAnnotationMethodUncached(XFieldElement fieldElement) { - return findProperty(fieldElement) - .methodForAnnotationsSignature() + return Optional.ofNullable(findProperty(fieldElement).getMethodForAnnotationsSignature()) .map( signature -> Optional.ofNullable(methodDescriptors().get(signature)) @@ -120,20 +102,19 @@ Optional getPropertyGetter(XFieldElement fieldElement) { } private Optional getPropertyGetterUncached(XFieldElement fieldElement) { - return findProperty(fieldElement) - .getterSignature() + return Optional.ofNullable(findProperty(fieldElement).getGetterSignature()) .flatMap(signature -> Optional.ofNullable(methodDescriptors().get(signature))); } private PropertyMetadata findProperty(XFieldElement field) { String fieldDescriptor = field.getJvmDescriptor(); - if (classMetadata().propertiesByFieldSignature().containsKey(fieldDescriptor)) { - return classMetadata().propertiesByFieldSignature().get(fieldDescriptor); + if (classMetadata().getPropertiesBySignature().containsKey(fieldDescriptor)) { + return classMetadata().getPropertiesBySignature().get(fieldDescriptor); } else { // Fallback to finding property by name, see: https://youtrack.jetbrains.com/issue/KT-35124 final String propertyName = getPropertyNameFromField(field); - return classMetadata().propertiesByFieldSignature().values().stream() - .filter(property -> propertyName.contentEquals(property.name())) + return classMetadata().getPropertiesBySignature().values().stream() + .filter(property -> propertyName.contentEquals(property.getName())) // SUPPRESS_GET_NAME_CHECK .collect(DaggerCollectors.onlyElement()); } } @@ -149,204 +130,7 @@ private static String getPropertyNameFromField(XFieldElement field) { /** Parse Kotlin class metadata from a given type element. */ static KotlinMetadata from(XTypeElement typeElement) { - return new AutoValue_KotlinMetadata(typeElement, ClassMetadata.create(metadataOf(typeElement))); - } - - private static KotlinClassMetadata.Class metadataOf(XTypeElement typeElement) { - XAnnotation annotation = typeElement.getAnnotation(ClassNames.KOTLIN_METADATA); - Metadata metadataAnnotation = - JvmMetadataUtil.Metadata( - annotation.getAsInt("k"), - annotation.getAsIntList("mv").stream().mapToInt(Integer::intValue).toArray(), - annotation.getAsStringList("d1").toArray(new String[0]), - annotation.getAsStringList("d2").toArray(new String[0]), - annotation.getAsString("xs"), - getOptionalStringValue(annotation, "pn").orElse(null), - getOptionalIntValue(annotation, "xi").orElse(null)); - KotlinClassMetadata metadata = KotlinClassMetadata.read(metadataAnnotation); - if (metadata == null) { - // Can happen if Kotlin < 1.0 or if metadata version is not supported, i.e. - // kotlinx-metadata-jvm is outdated. - throw new IllegalStateException( - "Unable to read Kotlin metadata due to unsupported metadata version."); - } - if (metadata instanceof KotlinClassMetadata.Class) { - // TODO(danysantiago): If when we need other types of metadata then move to right method. - return (KotlinClassMetadata.Class) metadata; - } else { - throw new IllegalStateException("Unsupported metadata type: " + metadata); - } - } - - @AutoValue - abstract static class ClassMetadata extends BaseMetadata { - abstract Optional companionObjectName(); - - abstract ImmutableSet constructors(); - - abstract ImmutableMap functionsBySignature(); - - abstract ImmutableMap propertiesByFieldSignature(); - - static ClassMetadata create(KotlinClassMetadata.Class metadata) { - KmClass kmClass = metadata.toKmClass(); - ClassMetadata.Builder builder = ClassMetadata.builder(kmClass.getFlags(), kmClass.getName()); - builder.companionObjectName(Optional.ofNullable(kmClass.getCompanionObject())); - kmClass.getConstructors().forEach(it -> builder.addConstructor(FunctionMetadata.create(it))); - kmClass.getFunctions().forEach(it -> builder.addFunction(FunctionMetadata.create(it))); - kmClass.getProperties().forEach(it -> builder.addProperty(PropertyMetadata.create(it))); - return builder.build(); - } - - private static Builder builder(int flags, String name) { - return new AutoValue_KotlinMetadata_ClassMetadata.Builder().flags(flags).name(name); - } - - @AutoValue.Builder - abstract static class Builder implements BaseMetadata.Builder { - abstract Builder companionObjectName(Optional companionObjectName); - - abstract ImmutableSet.Builder constructorsBuilder(); - - abstract ImmutableMap.Builder functionsBySignatureBuilder(); - - abstract ImmutableMap.Builder propertiesByFieldSignatureBuilder(); - - Builder addConstructor(FunctionMetadata constructor) { - constructorsBuilder().add(constructor); - functionsBySignatureBuilder().put(constructor.signature(), constructor); - return this; - } - - Builder addFunction(FunctionMetadata function) { - functionsBySignatureBuilder().put(function.signature(), function); - return this; - } - - Builder addProperty(PropertyMetadata property) { - if (property.fieldSignature().isPresent()) { - propertiesByFieldSignatureBuilder().put(property.fieldSignature().get(), property); - } - return this; - } - - abstract ClassMetadata build(); - } - } - - @AutoValue - abstract static class FunctionMetadata extends BaseMetadata { - abstract String signature(); - - abstract ImmutableList parameters(); - - static FunctionMetadata create(KmConstructor metadata) { - FunctionMetadata.Builder builder = FunctionMetadata.builder(metadata.getFlags(), ""); - metadata - .getValueParameters() - .forEach( - it -> - builder.addParameter(ValueParameterMetadata.create(it.getFlags(), it.getName()))); - builder.signature(Objects.requireNonNull(JvmExtensionsKt.getSignature(metadata)).asString()); - return builder.build(); - } - - static FunctionMetadata create(KmFunction metadata) { - FunctionMetadata.Builder builder = - FunctionMetadata.builder(metadata.getFlags(), metadata.getName()); - metadata - .getValueParameters() - .forEach( - it -> - builder.addParameter(ValueParameterMetadata.create(it.getFlags(), it.getName()))); - builder.signature(Objects.requireNonNull(JvmExtensionsKt.getSignature(metadata)).asString()); - return builder.build(); - } - - private static Builder builder(int flags, String name) { - return new AutoValue_KotlinMetadata_FunctionMetadata.Builder().flags(flags).name(name); - } - - @AutoValue.Builder - abstract static class Builder implements BaseMetadata.Builder { - abstract Builder signature(String signature); - - abstract ImmutableList.Builder parametersBuilder(); - - Builder addParameter(ValueParameterMetadata parameter) { - parametersBuilder().add(parameter); - return this; - } - - abstract FunctionMetadata build(); - } - } - - @AutoValue - abstract static class PropertyMetadata extends BaseMetadata { - /** Returns the JVM field descriptor of the backing field of this property. */ - abstract Optional fieldSignature(); - - abstract Optional getterSignature(); - - /** Returns JVM method descriptor of the synthetic method for property annotations. */ - abstract Optional methodForAnnotationsSignature(); - - static PropertyMetadata create(KmProperty metadata) { - PropertyMetadata.Builder builder = - PropertyMetadata.builder(metadata.getFlags(), metadata.getName()); - builder.fieldSignature( - Optional.ofNullable(JvmExtensionsKt.getFieldSignature(metadata)) - .map(JvmFieldSignature::asString)); - builder.getterSignature( - Optional.ofNullable(JvmExtensionsKt.getGetterSignature(metadata)) - .map(JvmMethodSignature::asString)); - builder.methodForAnnotationsSignature( - Optional.ofNullable(JvmExtensionsKt.getSyntheticMethodForAnnotations(metadata)) - .map(JvmMethodSignature::asString)); - return builder.build(); - } - - private static Builder builder(int flags, String name) { - return new AutoValue_KotlinMetadata_PropertyMetadata.Builder().flags(flags).name(name); - } - - @AutoValue.Builder - interface Builder extends BaseMetadata.Builder { - Builder fieldSignature(Optional signature); - - Builder getterSignature(Optional signature); - - Builder methodForAnnotationsSignature(Optional signature); - - PropertyMetadata build(); - } - } - - @AutoValue - abstract static class ValueParameterMetadata extends BaseMetadata { - private static ValueParameterMetadata create(int flags, String name) { - return new AutoValue_KotlinMetadata_ValueParameterMetadata(flags, name); - } - } - - abstract static class BaseMetadata { - /** Returns the Kotlin metadata flags for this property. */ - abstract int flags(); - - /** returns {@code true} if the given flag (e.g. {@link Flag.IS_PRIVATE}) applies. */ - boolean flags(Flag flag) { - return flag.invoke(flags()); - } - - /** Returns the simple name of this property. */ - abstract String name(); - - interface Builder { - BuilderT flags(int flags); - - BuilderT name(String name); - } + return new AutoValue_KotlinMetadata(typeElement, ClassMetadata.of(typeElement)); } @AutoValue @@ -360,21 +144,4 @@ static MethodForAnnotations create(XMethodElement method) { @Nullable abstract XMethodElement method(); } - - private static Optional getOptionalIntValue(XAnnotation annotation, String valueName) { - return isValuePresent(annotation, valueName) - ? Optional.of(annotation.getAsInt(valueName)) - : Optional.empty(); - } - - private static Optional getOptionalStringValue(XAnnotation annotation, String valueName) { - return isValuePresent(annotation, valueName) - ? Optional.of(annotation.getAsString(valueName)) - : Optional.empty(); - } - - private static boolean isValuePresent(XAnnotation annotation, String valueName) { - return annotation.getAnnotationValues().stream() - .anyMatch(member -> member.getName().equals(valueName)); - } } diff --git a/java/dagger/internal/codegen/kotlin/BUILD b/java/dagger/internal/codegen/kotlin/BUILD index 41b82ab5078..bc2caf4bb9e 100644 --- a/java/dagger/internal/codegen/kotlin/BUILD +++ b/java/dagger/internal/codegen/kotlin/BUILD @@ -15,13 +15,16 @@ # Description: # Sources related to Kotlin metadata. -load("@rules_java//java:defs.bzl", "java_library") +load("@io_bazel_rules_kotlin//kotlin:jvm.bzl", "kt_jvm_library") package(default_visibility = ["//:src"]) -java_library( +kt_jvm_library( name = "kotlin", - srcs = glob(["*.java"]), + srcs = glob([ + "*.java", + "*.kt", + ]), plugins = ["//java/dagger/internal/codegen/bootstrap"], tags = ["maven:merged"], deps = [ diff --git a/java/dagger/internal/codegen/kotlin/ClassMetadata.kt b/java/dagger/internal/codegen/kotlin/ClassMetadata.kt new file mode 100644 index 00000000000..9ce0c56aa55 --- /dev/null +++ b/java/dagger/internal/codegen/kotlin/ClassMetadata.kt @@ -0,0 +1,93 @@ +/* + * 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.internal.codegen.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.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.signature +import kotlin.metadata.jvm.fieldSignature +import kotlin.metadata.jvm.getterSignature +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) }) + }.associate { it.signature to it } + + val propertiesBySignature = + kmClass.properties + .map { PropertyMetadata(it) } + .associate { it.fieldSignature to it } + .filterKeys { it != null } + + 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 +} diff --git a/java/dagger/internal/codegen/kotlin/KotlinMetadata.java b/java/dagger/internal/codegen/kotlin/KotlinMetadata.java index ef8f9a08e98..b06fd728a7d 100644 --- a/java/dagger/internal/codegen/kotlin/KotlinMetadata.java +++ b/java/dagger/internal/codegen/kotlin/KotlinMetadata.java @@ -19,35 +19,18 @@ import static dagger.internal.codegen.extension.DaggerStreams.toImmutableMap; import static dagger.internal.codegen.xprocessing.XElements.getSimpleName; -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 com.google.auto.value.AutoValue; import com.google.auto.value.extension.memoized.Memoized; -import com.google.common.base.Preconditions; -import com.google.common.collect.ImmutableList; import com.google.common.collect.ImmutableMap; -import com.google.common.collect.ImmutableSet; import dagger.internal.codegen.extension.DaggerCollectors; -import dagger.internal.codegen.javapoet.TypeNames; import java.util.HashMap; import java.util.Map; -import java.util.Objects; import java.util.Optional; import java.util.function.Function; import javax.annotation.Nullable; -import kotlin.Metadata; -import kotlin.metadata.Flag; -import kotlin.metadata.KmClass; -import kotlin.metadata.KmConstructor; -import kotlin.metadata.KmFunction; -import kotlin.metadata.KmProperty; -import kotlin.metadata.jvm.JvmExtensionsKt; -import kotlin.metadata.jvm.JvmFieldSignature; -import kotlin.metadata.jvm.JvmMetadataUtil; -import kotlin.metadata.jvm.JvmMethodSignature; -import kotlin.metadata.jvm.KotlinClassMetadata; /** Data class of a TypeElement and its Kotlin metadata. */ @AutoValue @@ -107,8 +90,7 @@ private Optional getAnnotationMethod(XFieldElement fieldEl } private Optional getAnnotationMethodUncached(XFieldElement fieldElement) { - return findProperty(fieldElement) - .methodForAnnotationsSignature() + return Optional.ofNullable(findProperty(fieldElement).getMethodForAnnotationsSignature()) .map( signature -> Optional.ofNullable(methodDescriptors().get(signature)) @@ -125,20 +107,19 @@ Optional getPropertyGetter(XFieldElement fieldElement) { } private Optional getPropertyGetterUncached(XFieldElement fieldElement) { - return findProperty(fieldElement) - .getterSignature() + return Optional.ofNullable(findProperty(fieldElement).getGetterSignature()) .flatMap(signature -> Optional.ofNullable(methodDescriptors().get(signature))); } private PropertyMetadata findProperty(XFieldElement field) { String fieldDescriptor = field.getJvmDescriptor(); - if (classMetadata().propertiesByFieldSignature().containsKey(fieldDescriptor)) { - return classMetadata().propertiesByFieldSignature().get(fieldDescriptor); + if (classMetadata().getPropertiesBySignature().containsKey(fieldDescriptor)) { + return classMetadata().getPropertiesBySignature().get(fieldDescriptor); } else { // Fallback to finding property by name, see: https://youtrack.jetbrains.com/issue/KT-35124 final String propertyName = getPropertyNameFromField(field); - return classMetadata().propertiesByFieldSignature().values().stream() - .filter(property -> propertyName.contentEquals(property.name())) + return classMetadata().getPropertiesBySignature().values().stream() + .filter(property -> propertyName.contentEquals(property.getName())) // SUPPRESS_GET_NAME_CHECK .collect(DaggerCollectors.onlyElement()); } } @@ -154,217 +135,7 @@ private static String getPropertyNameFromField(XFieldElement field) { /** Parse Kotlin class metadata from a given type element. */ static KotlinMetadata from(XTypeElement typeElement) { - return new AutoValue_KotlinMetadata(typeElement, ClassMetadata.create(metadataOf(typeElement))); - } - - private static KotlinClassMetadata.Class metadataOf(XTypeElement typeElement) { - XAnnotation annotationMirror = typeElement.getAnnotation(TypeNames.KOTLIN_METADATA); - Preconditions.checkNotNull(annotationMirror); - Metadata metadataAnnotation = - JvmMetadataUtil.Metadata( - annotationMirror.getAsInt("k"), - annotationMirror.getAsIntList("mv").stream().mapToInt(Integer::intValue).toArray(), - annotationMirror.getAsStringList("d1").toArray(new String[0]), - annotationMirror.getAsStringList("d2").toArray(new String[0]), - annotationMirror.getAsString("xs"), - annotationMirror.getAnnotationValue("pn").hasStringValue() - ? annotationMirror.getAsString("pn") - : null, - annotationMirror.getAnnotationValue("xi").hasIntValue() - ? annotationMirror.getAsInt("xi") - : null); - KotlinClassMetadata metadata = KotlinClassMetadata.read(metadataAnnotation); - if (metadata == null) { - // Can happen if Kotlin < 1.0 or if metadata version is not supported, i.e. - // kotlinx-metadata-jvm is outdated. - throw new IllegalStateException( - "Unable to read Kotlin metadata due to unsupported metadata version."); - } - if (metadata instanceof KotlinClassMetadata.Class) { - // TODO(danysantiago): If when we need other types of metadata then move to right method. - return (KotlinClassMetadata.Class) metadata; - } else { - throw new IllegalStateException("Unsupported metadata type: " + metadata); - } - } - - @AutoValue - abstract static class ClassMetadata extends BaseMetadata { - abstract Optional companionObjectName(); - - abstract ImmutableSet constructors(); - - abstract ImmutableMap functionsBySignature(); - - abstract ImmutableMap propertiesByFieldSignature(); - - static ClassMetadata create(KotlinClassMetadata.Class metadata) { - KmClass kmClass = metadata.toKmClass(); - ClassMetadata.Builder builder = - ClassMetadata.builder( - kmClass.getFlags(), kmClass.getName()); // // SUPPRESS_GET_NAME_CHECK - builder.companionObjectName(Optional.ofNullable(kmClass.getCompanionObject())); - kmClass.getConstructors().forEach(it -> builder.addConstructor(FunctionMetadata.create(it))); - kmClass.getFunctions().forEach(it -> builder.addFunction(FunctionMetadata.create(it))); - kmClass.getProperties().forEach(it -> builder.addProperty(PropertyMetadata.create(it))); - return builder.build(); - } - - private static Builder builder(int flags, String name) { - return new AutoValue_KotlinMetadata_ClassMetadata.Builder().flags(flags).name(name); - } - - @AutoValue.Builder - abstract static class Builder implements BaseMetadata.Builder { - abstract Builder companionObjectName(Optional companionObjectName); - - abstract ImmutableSet.Builder constructorsBuilder(); - - abstract ImmutableMap.Builder functionsBySignatureBuilder(); - - abstract ImmutableMap.Builder propertiesByFieldSignatureBuilder(); - - Builder addConstructor(FunctionMetadata constructor) { - constructorsBuilder().add(constructor); - functionsBySignatureBuilder().put(constructor.signature(), constructor); - return this; - } - - Builder addFunction(FunctionMetadata function) { - functionsBySignatureBuilder().put(function.signature(), function); - return this; - } - - Builder addProperty(PropertyMetadata property) { - if (property.fieldSignature().isPresent()) { - propertiesByFieldSignatureBuilder().put(property.fieldSignature().get(), property); - } - return this; - } - - abstract ClassMetadata build(); - } - } - - @AutoValue - abstract static class FunctionMetadata extends BaseMetadata { - abstract String signature(); - - abstract ImmutableList parameters(); - - static FunctionMetadata create(KmConstructor metadata) { - FunctionMetadata.Builder builder = FunctionMetadata.builder(metadata.getFlags(), ""); - metadata - .getValueParameters() - .forEach( - it -> - builder.addParameter( - ValueParameterMetadata.create( - it.getFlags(), it.getName()))); // SUPPRESS_GET_NAME_CHECK - builder.signature(Objects.requireNonNull(JvmExtensionsKt.getSignature(metadata)).asString()); - return builder.build(); - } - - static FunctionMetadata create(KmFunction metadata) { - FunctionMetadata.Builder builder = - FunctionMetadata.builder( - metadata.getFlags(), metadata.getName()); // SUPPRESS_GET_NAME_CHECK - metadata - .getValueParameters() - .forEach( - it -> - builder.addParameter( - ValueParameterMetadata.create( - it.getFlags(), it.getName()))); // SUPPRESS_GET_NAME_CHECK - builder.signature(Objects.requireNonNull(JvmExtensionsKt.getSignature(metadata)).asString()); - return builder.build(); - } - - private static Builder builder(int flags, String name) { - return new AutoValue_KotlinMetadata_FunctionMetadata.Builder().flags(flags).name(name); - } - - @AutoValue.Builder - abstract static class Builder implements BaseMetadata.Builder { - abstract Builder signature(String signature); - - abstract ImmutableList.Builder parametersBuilder(); - - Builder addParameter(ValueParameterMetadata parameter) { - parametersBuilder().add(parameter); - return this; - } - - abstract FunctionMetadata build(); - } - } - - @AutoValue - abstract static class PropertyMetadata extends BaseMetadata { - /** Returns the JVM field descriptor of the backing field of this property. */ - abstract Optional fieldSignature(); - - abstract Optional getterSignature(); - - /** Returns JVM method descriptor of the synthetic method for property annotations. */ - abstract Optional methodForAnnotationsSignature(); - - static PropertyMetadata create(KmProperty metadata) { - PropertyMetadata.Builder builder = - PropertyMetadata.builder( - metadata.getFlags(), metadata.getName()); // SUPPRESS_GET_NAME_CHECK - builder.fieldSignature( - Optional.ofNullable(JvmExtensionsKt.getFieldSignature(metadata)) - .map(JvmFieldSignature::asString)); - builder.getterSignature( - Optional.ofNullable(JvmExtensionsKt.getGetterSignature(metadata)) - .map(JvmMethodSignature::asString)); - builder.methodForAnnotationsSignature( - Optional.ofNullable(JvmExtensionsKt.getSyntheticMethodForAnnotations(metadata)) - .map(JvmMethodSignature::asString)); - return builder.build(); - } - - private static Builder builder(int flags, String name) { - return new AutoValue_KotlinMetadata_PropertyMetadata.Builder().flags(flags).name(name); - } - - @AutoValue.Builder - interface Builder extends BaseMetadata.Builder { - Builder fieldSignature(Optional signature); - - Builder getterSignature(Optional signature); - - Builder methodForAnnotationsSignature(Optional signature); - - PropertyMetadata build(); - } - } - - @AutoValue - abstract static class ValueParameterMetadata extends BaseMetadata { - private static ValueParameterMetadata create(int flags, String name) { - return new AutoValue_KotlinMetadata_ValueParameterMetadata(flags, name); - } - } - - abstract static class BaseMetadata { - /** Returns the Kotlin metadata flags for this property. */ - abstract int flags(); - - /** returns {@code true} if the given flag (e.g. {@link Flag.IS_PRIVATE}) applies. */ - boolean flags(Flag flag) { - return flag.invoke(flags()); - } - - /** Returns the simple name of this property. */ - abstract String name(); - - interface Builder { - BuilderT flags(int flags); - - BuilderT name(String name); - } + return new AutoValue_KotlinMetadata(typeElement, ClassMetadata.of(typeElement)); } @AutoValue diff --git a/java/dagger/internal/codegen/kotlin/KotlinMetadataUtil.java b/java/dagger/internal/codegen/kotlin/KotlinMetadataUtil.java index 991b81ab4b6..9f80b3b10a6 100644 --- a/java/dagger/internal/codegen/kotlin/KotlinMetadataUtil.java +++ b/java/dagger/internal/codegen/kotlin/KotlinMetadataUtil.java @@ -29,7 +29,6 @@ import com.google.common.collect.ImmutableSet; import com.squareup.javapoet.ClassName; import dagger.internal.codegen.javapoet.TypeNames; -import dagger.internal.codegen.kotlin.KotlinMetadata.FunctionMetadata; import java.util.Optional; import javax.inject.Inject; @@ -86,7 +85,12 @@ public Optional getPropertyGetter(XFieldElement fieldElement) { public ImmutableMap getAllMethodNamesBySignature(XTypeElement element) { checkState( hasMetadata(element), "Can not call getAllMethodNamesBySignature for non-Kotlin class"); - return metadataFactory.create(element).classMetadata().functionsBySignature().values().stream() - .collect(toImmutableMap(FunctionMetadata::signature, FunctionMetadata::name)); + return metadataFactory.create(element) + .classMetadata() + .getFunctionsBySignature().values().stream() + .collect( + toImmutableMap( + FunctionMetadata::getSignature, + FunctionMetadata::getName)); // SUPPRESS_GET_NAME_CHECK } } diff --git a/javatests/artifacts/dagger-android-ksp/build.gradle b/javatests/artifacts/dagger-android-ksp/build.gradle index c438e4187cf..f6757914b32 100644 --- a/javatests/artifacts/dagger-android-ksp/build.gradle +++ b/javatests/artifacts/dagger-android-ksp/build.gradle @@ -17,8 +17,8 @@ buildscript { ext { agp_version = "8.1.1" - kotlin_version = "1.9.24" - ksp_version = "$kotlin_version-1.0.20" + kotlin_version = "2.0.21" + ksp_version = "$kotlin_version-1.0.28" } repositories { google() diff --git a/javatests/artifacts/dagger-ksp/build.gradle b/javatests/artifacts/dagger-ksp/build.gradle index 1864e003030..63b03418204 100644 --- a/javatests/artifacts/dagger-ksp/build.gradle +++ b/javatests/artifacts/dagger-ksp/build.gradle @@ -17,8 +17,8 @@ buildscript { ext { dagger_version = "LOCAL-SNAPSHOT" - kotlin_version = "1.9.24" - ksp_version = "$kotlin_version-1.0.20" + kotlin_version = "2.0.21" + ksp_version = "$kotlin_version-1.0.28" junit_version = "4.13" truth_version = "1.0.1" } diff --git a/javatests/artifacts/dagger/build.gradle b/javatests/artifacts/dagger/build.gradle index db2b9c05627..6c72ef04ee5 100644 --- a/javatests/artifacts/dagger/build.gradle +++ b/javatests/artifacts/dagger/build.gradle @@ -17,8 +17,8 @@ buildscript { ext { dagger_version = "LOCAL-SNAPSHOT" - kotlin_version = "1.9.24" - ksp_version = "$kotlin_version-1.0.20" + kotlin_version = "2.0.21" + ksp_version = "$kotlin_version-1.0.28" junit_version = "4.13" truth_version = "1.0.1" } diff --git a/javatests/artifacts/dagger/lazyclasskey/build.gradle b/javatests/artifacts/dagger/lazyclasskey/build.gradle index 9fdb3428637..16982d496d0 100644 --- a/javatests/artifacts/dagger/lazyclasskey/build.gradle +++ b/javatests/artifacts/dagger/lazyclasskey/build.gradle @@ -18,7 +18,7 @@ buildscript { ext { dagger_version = 'LOCAL-SNAPSHOT' agp_version = "8.1.1" - kotlin_version = '1.9.24' + kotlin_version = '2.0.21' } repositories { google() diff --git a/javatests/artifacts/hilt-android/lazyclasskey/build.gradle b/javatests/artifacts/hilt-android/lazyclasskey/build.gradle index d284e2790b6..53ced2c3e47 100644 --- a/javatests/artifacts/hilt-android/lazyclasskey/build.gradle +++ b/javatests/artifacts/hilt-android/lazyclasskey/build.gradle @@ -18,7 +18,7 @@ buildscript { ext { hilt_version = 'LOCAL-SNAPSHOT' agp_version = "8.1.1" - kotlin_version = '1.9.24' + kotlin_version = '2.0.21' } repositories { google() diff --git a/javatests/artifacts/hilt-android/simple/build.gradle b/javatests/artifacts/hilt-android/simple/build.gradle index 376aca054b5..3dfb1b9c034 100644 --- a/javatests/artifacts/hilt-android/simple/build.gradle +++ b/javatests/artifacts/hilt-android/simple/build.gradle @@ -17,7 +17,7 @@ buildscript { ext { dagger_version = 'LOCAL-SNAPSHOT' - kotlin_version = '1.9.24' + kotlin_version = '2.0.21' agp_version = System.getenv('AGP_VERSION') ?: "8.1.1" } repositories { diff --git a/javatests/artifacts/hilt-android/simpleKotlin/build.gradle b/javatests/artifacts/hilt-android/simpleKotlin/build.gradle index 48409476bec..b109613571e 100644 --- a/javatests/artifacts/hilt-android/simpleKotlin/build.gradle +++ b/javatests/artifacts/hilt-android/simpleKotlin/build.gradle @@ -16,8 +16,8 @@ buildscript { ext { - kotlin_version = '1.9.24' - ksp_version = "$kotlin_version-1.0.20" + kotlin_version = '2.0.21' + ksp_version = "$kotlin_version-1.0.28" agp_version = System.getenv('AGP_VERSION') ?: "8.1.1" } repositories { diff --git a/javatests/artifacts/hilt-android/viewmodel/build.gradle b/javatests/artifacts/hilt-android/viewmodel/build.gradle index 13c72dc2002..cb70181735c 100644 --- a/javatests/artifacts/hilt-android/viewmodel/build.gradle +++ b/javatests/artifacts/hilt-android/viewmodel/build.gradle @@ -18,7 +18,7 @@ buildscript { ext { hilt_version = 'LOCAL-SNAPSHOT' agp_version = "8.1.1" - kotlin_version = '1.9.24' + kotlin_version = '2.0.21' } repositories { google()