21
21
22
22
import java .io .IOException ;
23
23
import java .io .ObjectInputStream ;
24
+ import java .io .Serializable ;
24
25
import java .lang .annotation .Annotation ;
25
26
import java .lang .reflect .Method ;
26
27
import java .lang .reflect .Modifier ;
27
28
import java .lang .reflect .Type ;
28
- import java .util .ArrayList ;
29
- import java .util .Iterator ;
30
- import java .util .Random ;
29
+ import java .util .*;
30
+
31
31
import net .bytebuddy .ByteBuddy ;
32
32
import net .bytebuddy .description .method .MethodDescription ;
33
33
import net .bytebuddy .description .modifier .SynchronizationState ;
39
39
import net .bytebuddy .implementation .Implementation ;
40
40
import net .bytebuddy .implementation .attribute .MethodAttributeAppender ;
41
41
import net .bytebuddy .matcher .ElementMatcher ;
42
+ import net .bytebuddy .utility .GraalImageCode ;
43
+ import net .bytebuddy .utility .RandomString ;
42
44
import org .mockito .codegen .InjectionBase ;
43
45
import org .mockito .exceptions .base .MockitoException ;
44
46
import org .mockito .internal .creation .bytebuddy .ByteBuddyCrossClassLoaderSerializationSupport .CrossClassLoaderSerializableMock ;
@@ -52,7 +54,6 @@ class SubclassBytecodeGenerator implements BytecodeGenerator {
52
54
private final SubclassLoader loader ;
53
55
private final ModuleHandler handler ;
54
56
private final ByteBuddy byteBuddy ;
55
- private final Random random ;
56
57
private final Implementation readReplace ;
57
58
private final ElementMatcher <? super MethodDescription > matcher ;
58
59
@@ -82,8 +83,7 @@ protected SubclassBytecodeGenerator(
82
83
this .readReplace = readReplace ;
83
84
this .matcher = matcher ;
84
85
byteBuddy = new ByteBuddy ().with (TypeValidation .DISABLED );
85
- random = new Random ();
86
- handler = ModuleHandler .make (byteBuddy , loader , random );
86
+ handler = ModuleHandler .make (byteBuddy , loader );
87
87
}
88
88
89
89
private static boolean needsSamePackageClassLoader (MockFeatures <?> features ) {
@@ -167,7 +167,8 @@ public <T> Class<? extends T> mockClass(MockFeatures<T> features) {
167
167
&& features .serializableMode != SerializableMode .ACROSS_CLASSLOADERS
168
168
&& !isComingFromJDK (features .mockedType )
169
169
&& (loader .isDisrespectingOpenness ()
170
- || handler .isOpened (features .mockedType , MockAccess .class ));
170
+ || handler .isOpened (features .mockedType , MockAccess .class ))
171
+ && !GraalImageCode .getCurrent ().isDefined ();
171
172
String typeName ;
172
173
if (localMock
173
174
|| (loader instanceof MultipleParentClassLoader
@@ -180,7 +181,13 @@ public <T> Class<? extends T> mockClass(MockFeatures<T> features) {
180
181
+ features .mockedType .getSimpleName ();
181
182
}
182
183
String name =
183
- String .format ("%s$%s$%d" , typeName , "MockitoMock" , Math .abs (random .nextInt ()));
184
+ String .format (
185
+ "%s$%s$%s" ,
186
+ typeName ,
187
+ "MockitoMock" ,
188
+ GraalImageCode .getCurrent ().isDefined ()
189
+ ? suffix (features )
190
+ : RandomString .make ());
184
191
185
192
if (localMock ) {
186
193
handler .adjustModuleGraph (features .mockedType , MockAccess .class , false , true );
@@ -214,17 +221,36 @@ public <T> Class<? extends T> mockClass(MockFeatures<T> features) {
214
221
}
215
222
}
216
223
}
217
-
224
+ // Graal requires that the byte code of classes is identical what requires that interfaces
225
+ // are always
226
+ // defined in the exact same order. Therefore, we add an interface to the interface set if
227
+ // not mocking
228
+ // a class when Graal is active.
229
+ @ SuppressWarnings ("unchecked" )
230
+ Class <T > target =
231
+ GraalImageCode .getCurrent ().isDefined () && features .mockedType .isInterface ()
232
+ ? (Class <T >) Object .class
233
+ : features .mockedType ;
218
234
DynamicType .Builder <T > builder =
219
235
byteBuddy
220
- .subclass (features . mockedType )
236
+ .subclass (target )
221
237
.name (name )
222
238
.ignoreAlso (BytecodeGenerator .isGroovyMethod (false ))
223
239
.annotateType (
224
- features .stripAnnotations
240
+ features .stripAnnotations || features . mockedType . isInterface ()
225
241
? new Annotation [0 ]
226
242
: features .mockedType .getAnnotations ())
227
- .implement (new ArrayList <Type >(features .interfaces ))
243
+ .implement (
244
+ new ArrayList <>(
245
+ GraalImageCode .getCurrent ().isDefined ()
246
+ ? sortedSerializable (
247
+ features .interfaces ,
248
+ GraalImageCode .getCurrent ().isDefined ()
249
+ && features .mockedType
250
+ .isInterface ()
251
+ ? features .mockedType
252
+ : void .class )
253
+ : features .interfaces ))
228
254
.method (matcher )
229
255
.intercept (dispatcher )
230
256
.transform (withModifiers (SynchronizationState .PLAIN ))
@@ -266,6 +292,31 @@ public <T> Class<? extends T> mockClass(MockFeatures<T> features) {
266
292
.getLoaded ();
267
293
}
268
294
295
+ private static CharSequence suffix (MockFeatures <?> features ) {
296
+ // Constructs a deterministic suffix for this mock to assure that mocks always carry the
297
+ // same name.
298
+ StringBuilder sb = new StringBuilder ();
299
+ Set <String > names = new TreeSet <>();
300
+ names .add (features .mockedType .getName ());
301
+ for (Class <?> type : features .interfaces ) {
302
+ names .add (type .getName ());
303
+ }
304
+ return sb .append (RandomString .hashOf (names .hashCode ()))
305
+ .append (RandomString .hashOf (features .serializableMode .name ().hashCode ()))
306
+ .append (features .stripAnnotations ? "S" : "N" );
307
+ }
308
+
309
+ private static Collection <? extends Type > sortedSerializable (
310
+ Collection <Class <?>> interfaces , Class <?> mockedType ) {
311
+ SortedSet <Class <?>> types = new TreeSet <>(Comparator .comparing (Class ::getName ));
312
+ types .addAll (interfaces );
313
+ if (mockedType != void .class ) {
314
+ types .add (mockedType );
315
+ }
316
+ types .add (Serializable .class );
317
+ return types ;
318
+ }
319
+
269
320
@ Override
270
321
public void mockClassStatic (Class <?> type ) {
271
322
throw new MockitoException ("The subclass byte code generator cannot create static mocks" );
0 commit comments