Skip to content

Commit 6a5812b

Browse files
committed
Use primary copy method for Kotlin data classes.
We now resolve the copy method for Kotlin data classes that match the primary constructor. Previously, copy method resolution could find a secondary copy method as we didn't check for the primary constructor structure (parameter names and types). Closes #2324.
1 parent aaf5858 commit 6a5812b

File tree

4 files changed

+181
-8
lines changed

4 files changed

+181
-8
lines changed

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

+64-8
Original file line numberDiff line numberDiff line change
@@ -20,11 +20,13 @@
2020
import kotlin.reflect.KFunction;
2121
import kotlin.reflect.KParameter;
2222
import kotlin.reflect.KParameter.Kind;
23+
import kotlin.reflect.KType;
2324
import kotlin.reflect.full.KClasses;
2425
import kotlin.reflect.jvm.ReflectJvmMapping;
2526

2627
import java.lang.reflect.Method;
2728
import java.lang.reflect.Modifier;
29+
import java.lang.reflect.Type;
2830
import java.util.ArrayList;
2931
import java.util.Arrays;
3032
import java.util.List;
@@ -37,7 +39,8 @@
3739
import org.springframework.util.Assert;
3840

3941
/**
40-
* Value object to represent a Kotlin {@code copy} method.
42+
* Value object to represent a Kotlin {@code copy} method. The lookup requires a {@code copy} method that matches the
43+
* primary constructor of the class regardless of whether the primary constructor is the persistence constructor.
4144
*
4245
* @author Mark Paluch
4346
* @since 2.1
@@ -59,13 +62,14 @@ private KotlinCopyMethod(Method publicCopyMethod, Method syntheticCopyMethod) {
5962
this.publicCopyMethod = publicCopyMethod;
6063
this.syntheticCopyMethod = syntheticCopyMethod;
6164
this.copyFunction = ReflectJvmMapping.getKotlinFunction(publicCopyMethod);
62-
this.parameterCount = copyFunction.getParameters().size();
65+
this.parameterCount = copyFunction != null ? copyFunction.getParameters().size() : 0;
6366
}
6467

6568
/**
6669
* Attempt to lookup the Kotlin {@code copy} method. Lookup happens in two stages: Find the synthetic copy method and
6770
* then attempt to resolve its public variant.
6871
*
72+
* @param property the property that must be included in the copy method.
6973
* @param type the class.
7074
* @return {@link Optional} {@link KotlinCopyMethod}.
7175
*/
@@ -155,7 +159,6 @@ boolean shouldUsePublicCopyMethod(PersistentEntity<?, ?> entity) {
155159
return true;
156160
}
157161

158-
@SuppressWarnings("unchecked")
159162
private static Optional<Method> findPublicCopyMethod(Method defaultKotlinMethod) {
160163

161164
Class<?> type = defaultKotlinMethod.getDeclaringClass();
@@ -167,10 +170,7 @@ private static Optional<Method> findPublicCopyMethod(Method defaultKotlinMethod)
167170
return Optional.empty();
168171
}
169172

170-
List<KParameter> constructorArguments = primaryConstructor.getParameters() //
171-
.stream() //
172-
.filter(it -> it.getKind() == Kind.VALUE) //
173-
.collect(Collectors.toList());
173+
List<KParameter> constructorArguments = getComponentArguments(primaryConstructor);
174174

175175
return Arrays.stream(type.getDeclaredMethods()).filter(it -> it.getName().equals("copy") //
176176
&& !it.isSynthetic() //
@@ -207,7 +207,7 @@ private static boolean parameterMatches(List<KParameter> constructorArguments, K
207207

208208
KParameter constructorParameter = constructorArguments.get(constructorArgIndex);
209209

210-
if (!constructorParameter.getName().equals(parameter.getName())
210+
if (constructorParameter.getName() == null || !constructorParameter.getName().equals(parameter.getName())
211211
|| !constructorParameter.getType().equals(parameter.getType())) {
212212
return false;
213213
}
@@ -220,14 +220,70 @@ private static boolean parameterMatches(List<KParameter> constructorArguments, K
220220

221221
private static Optional<Method> findSyntheticCopyMethod(Class<?> type) {
222222

223+
KClass<?> kotlinClass = JvmClassMappingKt.getKotlinClass(type);
224+
KFunction<?> primaryConstructor = KClasses.getPrimaryConstructor(kotlinClass);
225+
226+
if (primaryConstructor == null) {
227+
return Optional.empty();
228+
}
229+
223230
return Arrays.stream(type.getDeclaredMethods()) //
224231
.filter(it -> it.getName().equals("copy$default") //
225232
&& Modifier.isStatic(it.getModifiers()) //
226233
&& it.getReturnType().equals(type))
227234
.filter(Method::isSynthetic) //
235+
.filter(it -> matchesPrimaryConstructor(it.getParameterTypes(), primaryConstructor))
228236
.findFirst();
229237
}
230238

239+
/**
240+
* Verify that the {@code parameterTypes} match arguments of the {@link KFunction primaryConstructor}.
241+
*/
242+
private static boolean matchesPrimaryConstructor(Class<?>[] parameterTypes, KFunction<?> primaryConstructor) {
243+
244+
List<KParameter> constructorArguments = getComponentArguments(primaryConstructor);
245+
246+
int defaultingArgs = KotlinDefaultMask.from(primaryConstructor, kParameter -> false).getDefaulting().length;
247+
248+
if (parameterTypes.length != 1 /* $this */ + constructorArguments.size() + defaultingArgs + 1 /* object marker */) {
249+
return false;
250+
}
251+
252+
// $this comes first
253+
if (!isAssignableFrom(parameterTypes[0], primaryConstructor.getReturnType())) {
254+
return false;
255+
}
256+
257+
for (int i = 0; i < constructorArguments.size(); i++) {
258+
259+
KParameter kParameter = constructorArguments.get(i);
260+
261+
if (!isAssignableFrom(parameterTypes[i + 1], kParameter.getType())) {
262+
return false;
263+
}
264+
}
265+
266+
return true;
267+
}
268+
269+
private static List<KParameter> getComponentArguments(KFunction<?> primaryConstructor) {
270+
return primaryConstructor.getParameters() //
271+
.stream() //
272+
.filter(it -> it.getKind() == Kind.VALUE) //
273+
.collect(Collectors.toList());
274+
}
275+
276+
private static boolean isAssignableFrom(Class<?> target, KType source) {
277+
278+
Type parameterType = ReflectJvmMapping.getJavaType(source);
279+
280+
if (parameterType instanceof Class) {
281+
return target.isAssignableFrom((Class<?>) parameterType);
282+
}
283+
284+
return false;
285+
}
286+
231287
/**
232288
* Value object to represent Kotlin {@literal copy$default} invocation metadata.
233289
*

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

+14
Original file line numberDiff line numberDiff line change
@@ -18,10 +18,12 @@
1818
import static org.assertj.core.api.Assertions.*;
1919

2020
import java.io.Serializable;
21+
import java.time.LocalDateTime;
2122

2223
import org.junit.jupiter.api.Test;
2324
import org.springframework.data.annotation.Id;
2425
import org.springframework.data.mapping.PersistentEntity;
26+
import org.springframework.data.mapping.PersistentPropertyAccessor;
2527
import org.springframework.data.mapping.context.SampleMappingContext;
2628
import org.springframework.data.mapping.context.SamplePersistentProperty;
2729
import org.springframework.data.repository.core.EntityInformation;
@@ -32,6 +34,7 @@
3234
*
3335
* @author John Blum
3436
* @author Oliver Gierke
37+
* @author Mark Paluch
3538
*/
3639
public class ClassGeneratingPropertyAccessorFactoryEntityTypeTests {
3740

@@ -53,6 +56,17 @@ public void getIdentifierOfClassBasedEntity() {
5356
assertThat(getEntityInformation(Person.class).getId(jonDoe)).isEqualTo(jonDoe.name);
5457
}
5558

59+
@Test // #2324
60+
public void shouldGeneratePropertyAccessorForKotlinClassWithMultipleCopyMethods() {
61+
62+
ClassGeneratingPropertyAccessorFactory factory = new ClassGeneratingPropertyAccessorFactory();
63+
PersistentPropertyAccessor<WithCustomCopyMethod> propertyAccessor = factory.getPropertyAccessor(
64+
mappingContext.getRequiredPersistentEntity(WithCustomCopyMethod.class),
65+
new WithCustomCopyMethod("", "", "", 1, LocalDateTime.MAX, LocalDateTime.MAX, ""));
66+
67+
assertThat(propertyAccessor).isNotNull();
68+
}
69+
5670
private EntityInformation<Object, Serializable> getEntityInformation(Class<?> type) {
5771

5872
PersistentEntity<Object, SamplePersistentProperty> entity = mappingContext.getRequiredPersistentEntity(type);
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,75 @@
1+
/*
2+
* Copyright 2021 the original author or authors.
3+
*
4+
* Licensed under the Apache License, Version 2.0 (the "License");
5+
* you may not use this file except in compliance with the License.
6+
* You may obtain a copy of the License at
7+
*
8+
* https://www.apache.org/licenses/LICENSE-2.0
9+
*
10+
* Unless required by applicable law or agreed to in writing, software
11+
* distributed under the License is distributed on an "AS IS" BASIS,
12+
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13+
* See the License for the specific language governing permissions and
14+
* limitations under the License.
15+
*/
16+
package org.springframework.data.mapping.model;
17+
18+
import static org.assertj.core.api.Assertions.*;
19+
20+
import java.util.Optional;
21+
22+
import org.junit.jupiter.api.Test;
23+
24+
import org.springframework.data.mapping.context.SampleMappingContext;
25+
26+
/**
27+
* Unit tests for {@link KotlinCopyMethod}.
28+
*
29+
* @author Mark Paluch
30+
*/
31+
class KotlinCopyMethodUnitTests {
32+
33+
SampleMappingContext mappingContext = new SampleMappingContext();
34+
35+
@Test // #2324
36+
void shouldLookupPrimaryConstructor() {
37+
38+
Optional<KotlinCopyMethod> optional = KotlinCopyMethod.findCopyMethod(DataClassKt.class);
39+
40+
assertThat(optional).hasValueSatisfying(actual -> {
41+
// $this, 1 component, 1 defaulting mask, 1 Object marker
42+
assertThat(actual.getSyntheticCopyMethod().getParameterCount()).isEqualTo(4);
43+
44+
// $this, 1 component
45+
assertThat(actual.getCopyFunction().getParameters()).hasSize(2);
46+
});
47+
}
48+
49+
@Test // #2324
50+
void shouldLookupPrimaryConstructorWhenTwoCopyMethodsArePresent() {
51+
52+
Optional<KotlinCopyMethod> optional = KotlinCopyMethod.findCopyMethod(WithCustomCopyMethod.class);
53+
54+
assertThat(optional).hasValueSatisfying(actual -> {
55+
// $this, 7 components, 1 defaulting mask, 1 Object marker
56+
assertThat(actual.getSyntheticCopyMethod().getParameterCount()).isEqualTo(10);
57+
58+
// $this, 7 components
59+
assertThat(actual.getCopyFunction().getParameters()).hasSize(8);
60+
61+
assertThat(
62+
actual.shouldUsePublicCopyMethod(mappingContext.getRequiredPersistentEntity(WithCustomCopyMethod.class)))
63+
.isFalse();
64+
});
65+
}
66+
67+
@Test // #2324
68+
void shouldUsePublicKotlinMethodForSinglePropertyEntities() {
69+
70+
KotlinCopyMethod copyMethod = KotlinCopyMethod.findCopyMethod(DataClassKt.class).get();
71+
72+
assertThat(copyMethod.shouldUsePublicCopyMethod(mappingContext.getRequiredPersistentEntity(DataClassKt.class)))
73+
.isTrue();
74+
}
75+
}

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

+28
Original file line numberDiff line numberDiff line change
@@ -15,6 +15,7 @@
1515
*/
1616
package org.springframework.data.mapping.model
1717

18+
import java.time.LocalDateTime
1819
import java.util.*
1920

2021
/**
@@ -31,3 +32,30 @@ data class ExtendedDataClassKt(val id: Long, val name: String) {
3132
data class SingleSettableProperty constructor(val id: UUID = UUID.randomUUID()) {
3233
val version: Int? = null
3334
}
35+
36+
data class WithCustomCopyMethod(
37+
val id: String?,
38+
val userId: String,
39+
val status: String,
40+
val attempts: Int,
41+
val createdAt: LocalDateTime,
42+
val updatedAt: LocalDateTime,
43+
val sessionId: String?
44+
) {
45+
46+
fun copy(
47+
status: String,
48+
updatedAt: LocalDateTime,
49+
sessionId: String,
50+
attempts: Int = this.attempts
51+
) = WithCustomCopyMethod(
52+
this.id,
53+
this.userId,
54+
status,
55+
attempts,
56+
this.createdAt,
57+
updatedAt,
58+
sessionId
59+
)
60+
61+
}

0 commit comments

Comments
 (0)