Skip to content

Commit 25c7c75

Browse files
christophstroblmp911de
authored andcommitted
Default generic type arguments when resolving KType from a Class.
We now fill up missing KTypeProjection arguments with star because the Kotlin Reflection.typeOf resolution fails if arguments are not provided. Closes #3041 Original pull request: #3048
1 parent 2ebb6a7 commit 25c7c75

File tree

2 files changed

+55
-1
lines changed

2 files changed

+55
-1
lines changed

src/main/java/org/springframework/data/mapping/model/KotlinValueUtils.java

+25-1
Original file line numberDiff line numberDiff line change
@@ -25,10 +25,12 @@
2525
import kotlin.reflect.KProperty;
2626
import kotlin.reflect.KType;
2727
import kotlin.reflect.KTypeParameter;
28+
import kotlin.reflect.KTypeProjection;
2829
import kotlin.reflect.jvm.ReflectJvmMapping;
2930

3031
import java.lang.reflect.Type;
3132
import java.util.ArrayList;
33+
import java.util.Arrays;
3234
import java.util.Collections;
3335
import java.util.List;
3436

@@ -39,6 +41,7 @@
3941
* Utilities for Kotlin Value class support.
4042
*
4143
* @author Mark Paluch
44+
* @author Christoph Strobl
4245
* @since 3.2
4346
*/
4447
class KotlinValueUtils {
@@ -72,7 +75,28 @@ public static ValueBoxing getConstructorValueHierarchy(KParameter parameter) {
7275
public static ValueBoxing getConstructorValueHierarchy(Class<?> cls) {
7376

7477
KClass<?> kotlinClass = JvmClassMappingKt.getKotlinClass(cls);
75-
return new ValueBoxing(BoxingRules.CONSTRUCTOR, Reflection.typeOf(kotlinClass), kotlinClass, false);
78+
KType kType = extractKType(kotlinClass);
79+
return new ValueBoxing(BoxingRules.CONSTRUCTOR, kType, kotlinClass, false);
80+
}
81+
82+
/**
83+
* Get the {@link KType} for a given {@link KClass} and potentially fill missing generic type arguments with
84+
* {@link KTypeProjection#star} to prevent Kotlin internal checks to fail.
85+
*
86+
* @param kotlinClass
87+
* @return
88+
*/
89+
private static KType extractKType(KClass<?> kotlinClass) {
90+
91+
return kotlinClass.getTypeParameters().isEmpty() ? Reflection.typeOf(kotlinClass)
92+
: Reflection.typeOf(JvmClassMappingKt.getJavaClass(kotlinClass), stubKTypeProjections(kotlinClass));
93+
}
94+
95+
private static KTypeProjection[] stubKTypeProjections(KClass<?> kotlinClass) {
96+
97+
KTypeProjection[] kTypeProjections = new KTypeProjection[kotlinClass.getTypeParameters().size()];
98+
Arrays.fill(kTypeProjections, KTypeProjection.star);
99+
return kTypeProjections;
76100
}
77101

78102
/**

src/test/kotlin/org/springframework/data/mapping/model/KotlinClassGeneratingEntityInstantiatorUnitTests.kt

+30
Original file line numberDiff line numberDiff line change
@@ -21,8 +21,11 @@ import org.assertj.core.api.Assertions.assertThat
2121
import org.assertj.core.api.Assertions.assertThatThrownBy
2222
import org.junit.jupiter.api.Test
2323
import org.springframework.data.annotation.PersistenceConstructor
24+
import org.springframework.data.annotation.Persistent
2425
import org.springframework.data.mapping.PersistentEntity
2526
import org.springframework.data.mapping.context.SamplePersistentProperty
27+
import org.springframework.data.mapping.model.KotlinValueUtils.BoxingRules
28+
import kotlin.jvm.internal.Reflection
2629
import kotlin.reflect.KClass
2730

2831
/**
@@ -149,6 +152,28 @@ class KotlinClassGeneratingEntityInstantiatorUnitTests {
149152
assertThat(instance.aBool).isTrue()
150153
}
151154

155+
@Test // GH-3041
156+
fun `should pick preferred constructor if multiple with same argument count are present`() {
157+
158+
val entity =
159+
mockk<PersistentEntity<WithConstructorsHavingSameParameterCount, SamplePersistentProperty>>()
160+
val constructor =
161+
PreferredConstructorDiscoverer.discover<WithConstructorsHavingSameParameterCount, SamplePersistentProperty>(
162+
WithConstructorsHavingSameParameterCount::class.java
163+
)
164+
165+
every { provider.getParameterValue<Any>(any()) }.returns(1L).andThen(null)
166+
every { entity.instanceCreatorMetadata } returns constructor
167+
every { entity.type } returns constructor!!.constructor.declaringClass
168+
every { entity.typeInformation } returns mockk()
169+
170+
val instance: WithConstructorsHavingSameParameterCount =
171+
KotlinClassGeneratingEntityInstantiator().createInstance(entity, provider)
172+
173+
assertThat(instance.id).isEqualTo(1L)
174+
assertThat(instance.notes).isEmpty();
175+
}
176+
152177
@Test // DATACMNS-1338
153178
fun `should create instance using @PersistenceConstructor`() {
154179

@@ -271,6 +296,11 @@ class KotlinClassGeneratingEntityInstantiatorUnitTests {
271296
val aFloat: Float = 0.0f, val aDouble: Double = 0.0, val aChar: Char = 'a', val aBool: Boolean = true
272297
)
273298

299+
300+
data class WithConstructorsHavingSameParameterCount @PersistenceConstructor constructor(val id: Long?, val notes: Map<String, String> = emptyMap()) {
301+
constructor(notes: Map<String, String>, additionalNotes: Map<String, String> = emptyMap()) : this(null, notes + additionalNotes)
302+
}
303+
274304
data class ContactWithPersistenceConstructor(val firstname: String, val lastname: String) {
275305

276306
@PersistenceConstructor

0 commit comments

Comments
 (0)