Skip to content

Commit b6581eb

Browse files
committed
Fix Kotlin copy method assignability check.
We now resolve only the raw class when checking if a primary constructor argument is assignable to method parameters of the synthetic copy method. Previously we used ResolvableType's assignability check which considered generic type arguments. As the type resolution between the KType and copy method type is not symmetric, the check only succeeded in cases where both types could be resolved to the same type/assignable type. Using projections or Any caused asymmetric resolution and therefor the assignability check returned non-assignable. Closes #2324.
1 parent 8b5f467 commit b6581eb

File tree

3 files changed

+14
-5
lines changed

3 files changed

+14
-5
lines changed

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

+2-1
Original file line numberDiff line numberDiff line change
@@ -278,7 +278,8 @@ private static boolean isAssignableFrom(Class<?> target, KType source) {
278278

279279
Type parameterType = ReflectJvmMapping.getJavaType(source);
280280

281-
return ResolvableType.forClass(target).isAssignableFrom(ResolvableType.forType(parameterType));
281+
Class<?> rawClass = ResolvableType.forType(parameterType).getRawClass();
282+
return rawClass == null || target.isAssignableFrom(rawClass);
282283
}
283284

284285
/**

src/test/java/org/springframework/data/mapping/model/KotlinCopyMethodUnitTests.java

+3-4
Original file line numberDiff line numberDiff line change
@@ -73,11 +73,10 @@ void shouldUsePublicKotlinMethodForSinglePropertyEntities() {
7373
.isTrue();
7474
}
7575

76-
@Test // #2324
76+
@Test // #2324, #2336
7777
void shouldDetermineCopyMethodForParametrizedType() {
7878

79-
Optional<KotlinCopyMethod> copyMethod = KotlinCopyMethod.findCopyMethod(ImmutableKotlinPerson.class);
80-
81-
assertThat(copyMethod).isPresent();
79+
assertThat(KotlinCopyMethod.findCopyMethod(ImmutableKotlinPerson.class)).isPresent();
80+
assertThat(KotlinCopyMethod.findCopyMethod(DataClassWithParametrizedCollections.class)).isPresent();
8281
}
8382
}

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

+9
Original file line numberDiff line numberDiff line change
@@ -65,3 +65,12 @@ data class ImmutableKotlinPerson(
6565
@Id val name: String,
6666
val wasOnboardedBy: List<ImmutableKotlinPerson>
6767
)
68+
69+
data class DataClassWithParametrizedCollections<T>(
70+
val id: String? = null,
71+
val flags: Map<out String, Any>,
72+
val stringStringFlags: Map<in String, String>,
73+
val parametrized: List<T>,
74+
val anyList: List<*>
75+
)
76+

0 commit comments

Comments
 (0)