20
20
import kotlin .reflect .KFunction ;
21
21
import kotlin .reflect .KParameter ;
22
22
import kotlin .reflect .KParameter .Kind ;
23
+ import kotlin .reflect .KType ;
23
24
import kotlin .reflect .full .KClasses ;
24
25
import kotlin .reflect .jvm .ReflectJvmMapping ;
25
26
26
27
import java .lang .reflect .Method ;
27
28
import java .lang .reflect .Modifier ;
29
+ import java .lang .reflect .Type ;
28
30
import java .util .ArrayList ;
29
31
import java .util .Arrays ;
30
32
import java .util .List ;
37
39
import org .springframework .util .Assert ;
38
40
39
41
/**
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.
41
44
*
42
45
* @author Mark Paluch
43
46
* @since 2.1
@@ -59,13 +62,14 @@ private KotlinCopyMethod(Method publicCopyMethod, Method syntheticCopyMethod) {
59
62
this .publicCopyMethod = publicCopyMethod ;
60
63
this .syntheticCopyMethod = syntheticCopyMethod ;
61
64
this .copyFunction = ReflectJvmMapping .getKotlinFunction (publicCopyMethod );
62
- this .parameterCount = copyFunction .getParameters ().size ();
65
+ this .parameterCount = copyFunction != null ? copyFunction .getParameters ().size () : 0 ;
63
66
}
64
67
65
68
/**
66
69
* Attempt to lookup the Kotlin {@code copy} method. Lookup happens in two stages: Find the synthetic copy method and
67
70
* then attempt to resolve its public variant.
68
71
*
72
+ * @param property the property that must be included in the copy method.
69
73
* @param type the class.
70
74
* @return {@link Optional} {@link KotlinCopyMethod}.
71
75
*/
@@ -155,7 +159,6 @@ boolean shouldUsePublicCopyMethod(PersistentEntity<?, ?> entity) {
155
159
return true ;
156
160
}
157
161
158
- @ SuppressWarnings ("unchecked" )
159
162
private static Optional <Method > findPublicCopyMethod (Method defaultKotlinMethod ) {
160
163
161
164
Class <?> type = defaultKotlinMethod .getDeclaringClass ();
@@ -167,10 +170,7 @@ private static Optional<Method> findPublicCopyMethod(Method defaultKotlinMethod)
167
170
return Optional .empty ();
168
171
}
169
172
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 );
174
174
175
175
return Arrays .stream (type .getDeclaredMethods ()).filter (it -> it .getName ().equals ("copy" ) //
176
176
&& !it .isSynthetic () //
@@ -207,7 +207,7 @@ private static boolean parameterMatches(List<KParameter> constructorArguments, K
207
207
208
208
KParameter constructorParameter = constructorArguments .get (constructorArgIndex );
209
209
210
- if (!constructorParameter .getName ().equals (parameter .getName ())
210
+ if (constructorParameter . getName () == null || !constructorParameter .getName ().equals (parameter .getName ())
211
211
|| !constructorParameter .getType ().equals (parameter .getType ())) {
212
212
return false ;
213
213
}
@@ -220,14 +220,70 @@ private static boolean parameterMatches(List<KParameter> constructorArguments, K
220
220
221
221
private static Optional <Method > findSyntheticCopyMethod (Class <?> type ) {
222
222
223
+ KClass <?> kotlinClass = JvmClassMappingKt .getKotlinClass (type );
224
+ KFunction <?> primaryConstructor = KClasses .getPrimaryConstructor (kotlinClass );
225
+
226
+ if (primaryConstructor == null ) {
227
+ return Optional .empty ();
228
+ }
229
+
223
230
return Arrays .stream (type .getDeclaredMethods ()) //
224
231
.filter (it -> it .getName ().equals ("copy$default" ) //
225
232
&& Modifier .isStatic (it .getModifiers ()) //
226
233
&& it .getReturnType ().equals (type ))
227
234
.filter (Method ::isSynthetic ) //
235
+ .filter (it -> matchesPrimaryConstructor (it .getParameterTypes (), primaryConstructor ))
228
236
.findFirst ();
229
237
}
230
238
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
+
231
287
/**
232
288
* Value object to represent Kotlin {@literal copy$default} invocation metadata.
233
289
*
0 commit comments