Skip to content

Commit d539af2

Browse files
raphwSteve Green
and
Steve Green
authored
Fix mockito-inline for Groovy projects (#2618)
Add ignore matcher for Groovy meta methods for inline Byte Buddy mock maker. Add a new test subproject to verify fixes between mockito-inline and Groovy Fixes #2522 Co-authored-by: Steve Green <[email protected]>
1 parent 2947e9d commit d539af2

File tree

6 files changed

+85
-26
lines changed

6 files changed

+85
-26
lines changed

settings.gradle.kts

+1
Original file line numberDiff line numberDiff line change
@@ -6,6 +6,7 @@ include("inline",
66
"proxy",
77
"extTest",
88
"groovyTest",
9+
"groovyInlineTest",
910
"kotlinTest",
1011
"kotlinReleaseCoroutinesTest",
1112
"android",

src/main/java/org/mockito/internal/creation/bytebuddy/BytecodeGenerator.java

+18
Original file line numberDiff line numberDiff line change
@@ -4,6 +4,12 @@
44
*/
55
package org.mockito.internal.creation.bytebuddy;
66

7+
import net.bytebuddy.description.method.MethodDescription;
8+
import net.bytebuddy.matcher.ElementMatcher;
9+
10+
import static net.bytebuddy.matcher.ElementMatchers.*;
11+
import static net.bytebuddy.matcher.ElementMatchers.named;
12+
713
public interface BytecodeGenerator {
814

915
<T> Class<? extends T> mockClass(MockFeatures<T> features);
@@ -13,4 +19,16 @@ public interface BytecodeGenerator {
1319
void mockClassStatic(Class<?> type);
1420

1521
default void clearAllCaches() {}
22+
23+
static ElementMatcher<MethodDescription> isGroovyMethod(boolean inline) {
24+
ElementMatcher.Junction<MethodDescription> matcher =
25+
isDeclaredBy(named("groovy.lang.GroovyObjectSupport"))
26+
.or(isAnnotatedWith(named("groovy.transform.Internal")));
27+
if (inline) {
28+
return matcher.or(
29+
named("$getStaticMetaClass").and(returns(named("groovy.lang.MetaClass"))));
30+
} else {
31+
return matcher;
32+
}
33+
}
1634
}

src/main/java/org/mockito/internal/creation/bytebuddy/InlineBytecodeGenerator.java

+8-4
Original file line numberDiff line numberDiff line change
@@ -117,15 +117,19 @@ public InlineBytecodeGenerator(
117117
.or(isEquals())
118118
.or(isDefaultFinalizer())))
119119
.and(
120-
not(
121-
isDeclaredBy(nameStartsWith("java."))
120+
not(isDeclaredBy(nameStartsWith("java."))
122121
.<MethodDescription>and(
123-
isPackagePrivate()))),
122+
isPackagePrivate()))
123+
.and(
124+
not(
125+
BytecodeGenerator
126+
.isGroovyMethod(
127+
true)))),
124128
Advice.withCustomMapping()
125129
.bind(MockMethodAdvice.Identifier.class, identifier)
126130
.to(MockMethodAdvice.class))
127131
.method(
128-
isStatic(),
132+
isStatic().and(not(BytecodeGenerator.isGroovyMethod(true))),
129133
Advice.withCustomMapping()
130134
.bind(MockMethodAdvice.Identifier.class, identifier)
131135
.to(MockMethodAdvice.ForStatic.class))

src/main/java/org/mockito/internal/creation/bytebuddy/SubclassBytecodeGenerator.java

+1-22
Original file line numberDiff line numberDiff line change
@@ -12,12 +12,9 @@
1212
import static net.bytebuddy.matcher.ElementMatchers.any;
1313
import static net.bytebuddy.matcher.ElementMatchers.hasParameters;
1414
import static net.bytebuddy.matcher.ElementMatchers.hasType;
15-
import static net.bytebuddy.matcher.ElementMatchers.isAnnotatedWith;
16-
import static net.bytebuddy.matcher.ElementMatchers.isDeclaredBy;
1715
import static net.bytebuddy.matcher.ElementMatchers.isEquals;
1816
import static net.bytebuddy.matcher.ElementMatchers.isHashCode;
1917
import static net.bytebuddy.matcher.ElementMatchers.isPackagePrivate;
20-
import static net.bytebuddy.matcher.ElementMatchers.named;
2118
import static net.bytebuddy.matcher.ElementMatchers.returns;
2219
import static net.bytebuddy.matcher.ElementMatchers.whereAny;
2320
import static org.mockito.internal.util.StringUtil.join;
@@ -29,9 +26,7 @@
2926
import java.lang.reflect.Modifier;
3027
import java.lang.reflect.Type;
3128
import java.util.ArrayList;
32-
import java.util.Collection;
3329
import java.util.Iterator;
34-
import java.util.LinkedList;
3530
import java.util.Random;
3631
import net.bytebuddy.ByteBuddy;
3732
import net.bytebuddy.description.method.MethodDescription;
@@ -224,7 +219,7 @@ public <T> Class<? extends T> mockClass(MockFeatures<T> features) {
224219
byteBuddy
225220
.subclass(features.mockedType)
226221
.name(name)
227-
.ignoreAlso(isGroovyMethod())
222+
.ignoreAlso(BytecodeGenerator.isGroovyMethod(false))
228223
.annotateType(
229224
features.stripAnnotations
230225
? new Annotation[0]
@@ -282,22 +277,6 @@ public void mockClassConstruction(Class<?> type) {
282277
"The subclass byte code generator cannot create construction mocks");
283278
}
284279

285-
private <T> Collection<Class<? super T>> getAllTypes(Class<T> type) {
286-
Collection<Class<? super T>> supertypes = new LinkedList<>();
287-
supertypes.add(type);
288-
Class<? super T> superType = type;
289-
while (superType != null) {
290-
supertypes.add(superType);
291-
superType = superType.getSuperclass();
292-
}
293-
return supertypes;
294-
}
295-
296-
private static ElementMatcher<MethodDescription> isGroovyMethod() {
297-
return isDeclaredBy(named("groovy.lang.GroovyObjectSupport"))
298-
.or(isAnnotatedWith(named("groovy.transform.Internal")));
299-
}
300-
301280
private boolean isComingFromJDK(Class<?> type) {
302281
// Comes from the manifest entry :
303282
// Implementation-Title: Java Runtime Environment
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,11 @@
1+
apply plugin: 'groovy'
2+
3+
description = "Integration test for using mockito-inline with Groovy."
4+
5+
apply from: "$rootDir/gradle/dependencies.gradle"
6+
7+
dependencies {
8+
testImplementation project(":inline")
9+
testImplementation libraries.groovy
10+
testImplementation libraries.junit4
11+
}
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,46 @@
1+
/*
2+
* Copyright (c) 2022 Mockito contributors
3+
* This program is made available under the terms of the MIT License.
4+
*/
5+
package org.mockito.groovy
6+
7+
import org.junit.Test
8+
import org.junit.runner.RunWith
9+
import org.mockito.InjectMocks
10+
import org.mockito.Mock
11+
import org.mockito.junit.MockitoJUnitRunner
12+
13+
import static org.mockito.Mockito.verify
14+
15+
@RunWith(MockitoJUnitRunner)
16+
class GroovyMockitoTest {
17+
18+
@Mock Helper helper
19+
@InjectMocks ClassUnderTest classUnderTest
20+
21+
/**
22+
* Test that the Groovy class under test can call methods on a mocked Groovy
23+
* helper class.
24+
*/
25+
@Test
26+
void testCallGroovyFromGroovy() {
27+
classUnderTest.methodUnderTest()
28+
verify(helper).helperMethod()
29+
}
30+
31+
static class ClassUnderTest {
32+
private final Helper helper
33+
34+
ClassUnderTest(Helper helper) {
35+
this.helper = helper
36+
}
37+
38+
void methodUnderTest() {
39+
helper.helperMethod()
40+
}
41+
}
42+
43+
static class Helper {
44+
void helperMethod() { }
45+
}
46+
}

0 commit comments

Comments
 (0)