Skip to content

Commit edd36d7

Browse files
committed
Log cause for failure to load AssumptionViolatedException
This commit improves diagnostics by logging the cause of a failure to load the AssumptionViolatedException class in OpenTest4JAndJUnit4AwareThrowableCollector. Issue: #2004
1 parent cc13950 commit edd36d7

File tree

2 files changed

+85
-17
lines changed

2 files changed

+85
-17
lines changed

junit-jupiter-engine/src/main/java/org/junit/jupiter/engine/support/OpenTest4JAndJUnit4AwareThrowableCollector.java

+17-3
Original file line numberDiff line numberDiff line change
@@ -12,6 +12,8 @@
1212

1313
import java.util.function.Predicate;
1414

15+
import org.junit.platform.commons.logging.Logger;
16+
import org.junit.platform.commons.logging.LoggerFactory;
1517
import org.junit.platform.commons.util.BlacklistedExceptions;
1618
import org.junit.platform.commons.util.ReflectionUtils;
1719
import org.junit.platform.engine.support.hierarchical.ThrowableCollector;
@@ -28,6 +30,13 @@
2830
*/
2931
class OpenTest4JAndJUnit4AwareThrowableCollector extends ThrowableCollector {
3032

33+
private static final Logger logger = LoggerFactory.getLogger(OpenTest4JAndJUnit4AwareThrowableCollector.class);
34+
35+
private static final String ASSUMPTION_VIOLATED_EXCEPTION = "org.junit.internal.AssumptionViolatedException";
36+
37+
private static final String COMMON_FAILURE_MESSAGE = "Failed to load class " + ASSUMPTION_VIOLATED_EXCEPTION
38+
+ ": only supporting " + TestAbortedException.class.getName() + " for aborted execution.";
39+
3140
private static final Predicate<? super Throwable> abortedExecutionPredicate = createAbortedExecutionPredicate();
3241

3342
OpenTest4JAndJUnit4AwareThrowableCollector() {
@@ -39,15 +48,20 @@ private static Predicate<? super Throwable> createAbortedExecutionPredicate() {
3948

4049
// Additionally support JUnit 4's AssumptionViolatedException?
4150
try {
42-
Class<?> clazz = ReflectionUtils.tryToLoadClass("org.junit.internal.AssumptionViolatedException").get();
51+
Class<?> clazz = ReflectionUtils.tryToLoadClass(ASSUMPTION_VIOLATED_EXCEPTION).get();
4352
if (clazz != null) {
4453
return otaPredicate.or(clazz::isInstance);
4554
}
4655
}
4756
catch (Throwable throwable) {
4857
BlacklistedExceptions.rethrowIfBlacklisted(throwable);
49-
// Otherwise ignore it since it's likely a ClassNotFoundException,
50-
// NoClassDefFoundError, or similar.
58+
if (throwable instanceof NoClassDefFoundError) {
59+
logger.info(throwable, () -> COMMON_FAILURE_MESSAGE + " Note that " + ASSUMPTION_VIOLATED_EXCEPTION
60+
+ " requires that Hamcrest is on the classpath.");
61+
}
62+
else {
63+
logger.debug(throwable, () -> COMMON_FAILURE_MESSAGE);
64+
}
5165
}
5266

5367
// Else just OTA's TestAbortedException

junit-jupiter-engine/src/test/java/org/junit/jupiter/engine/support/OpenTest4JAndJUnit4AwareThrowableCollectorTests.java

+68-14
Original file line numberDiff line numberDiff line change
@@ -10,52 +10,101 @@
1010

1111
package org.junit.jupiter.engine.support;
1212

13+
import static org.assertj.core.api.Assertions.assertThat;
1314
import static org.junit.jupiter.api.Assertions.assertNotNull;
1415
import static org.junit.jupiter.api.Assertions.assertThrows;
1516

1617
import java.net.URL;
1718
import java.net.URLClassLoader;
19+
import java.util.logging.Level;
20+
import java.util.logging.LogRecord;
1821

1922
import org.junit.internal.AssumptionViolatedException;
2023
import org.junit.jupiter.api.Test;
24+
import org.junit.jupiter.api.function.Executable;
25+
import org.junit.jupiter.engine.TrackLogRecords;
26+
import org.junit.platform.commons.logging.LogRecordListener;
2127
import org.junit.platform.commons.util.ReflectionUtils;
2228

2329
/**
2430
* Unit tests for {@link OpenTest4JAndJUnit4AwareThrowableCollector}.
2531
*
2632
* @since 5.5.2
2733
*/
34+
@TrackLogRecords
2835
class OpenTest4JAndJUnit4AwareThrowableCollectorTests {
2936

3037
@Test
31-
void simulateHamcrestNotInTheClasspath() throws Exception {
32-
ClassLoader originalClassLoader = Thread.currentThread().getContextClassLoader();
33-
try {
34-
HamcrestHidingClassLoader classLoader = new HamcrestHidingClassLoader();
38+
void simulateJUnit4NotInTheClasspath(LogRecordListener listener) throws Throwable {
39+
TestClassLoader classLoader = new TestClassLoader(true, false);
3540

36-
// We have to set our custom ClassLoader as the TCCL so that
37-
// ReflectionUtils uses it (indirectly via ClassLoaderUtils).
38-
Thread.currentThread().setContextClassLoader(classLoader);
41+
doWithCustomClassLoader(classLoader, () -> {
42+
// Ensure that our custom ClassLoader actually throws a ClassNotFoundException
43+
// when attempting to load the AssumptionViolatedException class.
44+
assertThrows(ClassNotFoundException.class,
45+
() -> ReflectionUtils.tryToLoadClass(AssumptionViolatedException.class.getName()).get());
46+
47+
Class<?> clazz = classLoader.loadClass(OpenTest4JAndJUnit4AwareThrowableCollector.class.getName());
48+
assertNotNull(ReflectionUtils.newInstance(clazz));
49+
50+
// @formatter:off
51+
assertThat(listener.stream(Level.FINE).map(LogRecord::getMessage).findFirst().orElse("<not found>"))
52+
.isEqualTo(
53+
"Failed to load class org.junit.internal.AssumptionViolatedException: " +
54+
"only supporting org.opentest4j.TestAbortedException for aborted execution.");
55+
// @formatter:on
56+
});
57+
}
58+
59+
@Test
60+
void simulateHamcrestNotInTheClasspath(LogRecordListener listener) throws Throwable {
61+
TestClassLoader classLoader = new TestClassLoader(false, true);
3962

63+
doWithCustomClassLoader(classLoader, () -> {
4064
// Ensure that our custom ClassLoader actually throws a NoClassDefFoundError
4165
// when attempting to load the AssumptionViolatedException class.
4266
assertThrows(NoClassDefFoundError.class,
43-
() -> ReflectionUtils.tryToLoadClass(AssumptionViolatedException.class.getName()));
67+
() -> ReflectionUtils.tryToLoadClass(AssumptionViolatedException.class.getName()).get());
4468

4569
Class<?> clazz = classLoader.loadClass(OpenTest4JAndJUnit4AwareThrowableCollector.class.getName());
4670
assertNotNull(ReflectionUtils.newInstance(clazz));
71+
72+
// @formatter:off
73+
assertThat(listener.stream(Level.INFO).map(LogRecord::getMessage).findFirst().orElse("<not found>"))
74+
.isEqualTo(
75+
"Failed to load class org.junit.internal.AssumptionViolatedException: " +
76+
"only supporting org.opentest4j.TestAbortedException for aborted execution. " +
77+
"Note that org.junit.internal.AssumptionViolatedException requires that Hamcrest is on the classpath.");
78+
// @formatter:on
79+
});
80+
}
81+
82+
private void doWithCustomClassLoader(ClassLoader classLoader, Executable executable) throws Throwable {
83+
ClassLoader originalClassLoader = Thread.currentThread().getContextClassLoader();
84+
try {
85+
// We have to set our custom ClassLoader as the TCCL so that
86+
// ReflectionUtils uses it (indirectly via ClassLoaderUtils).
87+
Thread.currentThread().setContextClassLoader(classLoader);
88+
89+
executable.execute();
4790
}
4891
finally {
4992
Thread.currentThread().setContextClassLoader(originalClassLoader);
5093
}
5194
}
5295

53-
private static class HamcrestHidingClassLoader extends URLClassLoader {
96+
private static class TestClassLoader extends URLClassLoader {
97+
98+
private static URL[] CLASSPATH_URLS = new URL[] {
99+
OpenTest4JAndJUnit4AwareThrowableCollector.class.getProtectionDomain().getCodeSource().getLocation() };
54100

55-
HamcrestHidingClassLoader() {
56-
super(new URL[] {
57-
OpenTest4JAndJUnit4AwareThrowableCollector.class.getProtectionDomain().getCodeSource().getLocation() },
58-
getSystemClassLoader());
101+
private final boolean simulateJUnit4Missing;
102+
private final boolean simulateHamcrestMissing;
103+
104+
public TestClassLoader(boolean simulateJUnit4Missing, boolean simulateHamcrestMissing) {
105+
super(CLASSPATH_URLS, getSystemClassLoader());
106+
this.simulateJUnit4Missing = simulateJUnit4Missing;
107+
this.simulateHamcrestMissing = simulateHamcrestMissing;
59108
}
60109

61110
@Override
@@ -66,8 +115,13 @@ public Class<?> loadClass(String name) throws ClassNotFoundException {
66115
return findClass(name);
67116
}
68117

118+
// Simulate that JUnit 4 is not in the classpath when loading AssumptionViolatedException
119+
if (this.simulateJUnit4Missing && name.equals(AssumptionViolatedException.class.getName())) {
120+
throw new ClassNotFoundException(AssumptionViolatedException.class.getName());
121+
}
122+
69123
// Simulate that Hamcrest is not in the classpath when loading AssumptionViolatedException
70-
if (name.equals(AssumptionViolatedException.class.getName())) {
124+
if (this.simulateHamcrestMissing && name.equals(AssumptionViolatedException.class.getName())) {
71125
throw new NoClassDefFoundError("org/hamcrest/SelfDescribing");
72126
}
73127

0 commit comments

Comments
 (0)