Skip to content

Commit e6d1614

Browse files
committed
Support automatic discovery of default TELs
Prior to this commit, there was no declarative mechanism for a custom TestExecutionListener to be registered as a default TestExecutionListener. This commit introduces support for discovering default TestExecutionListener implementations via the SpringFactoriesLoader mechanism. Specifically, the spring-test module declares all core default TestExecutionListeners under the org.springframework.test.context.TestExecutionListener key in its META-INF/spring.factories properties file, and third-party frameworks and developers can contribute to the list of default TestExecutionListeners in the same manner. - AbstractTestContextBootstrapper uses the SpringFactoriesLoader to look up the class names of all registered default TestExecutionListeners and sorts the instantiated listeners using AnnotationAwareOrderComparator. - DefaultTestContextBootstrapper and WebTestContextBootstrapper now rely on the SpringFactoriesLoader mechanism for finding default TestExecutionListeners instead of hard coding fully qualified class names. - To ensure that default TestExecutionListeners are registered in the correct order, each can implement Ordered or declare @order. - AbstractTestExecutionListener and all default TestExecutionListeners provided by Spring now implement Ordered with appropriate values. - Introduced "copy constructors" in MergedContextConfiguration and WebMergedContextConfiguration - SpringFactoriesLoader now uses AnnotationAwareOrderComparator instead of OrderComparator. Issue: SPR-11466
1 parent e142fd1 commit e6d1614

18 files changed

+227
-183
lines changed

spring-core/src/main/java/org/springframework/core/io/support/SpringFactoriesLoader.java

+5-4
Original file line numberDiff line numberDiff line change
@@ -27,7 +27,7 @@
2727
import org.apache.commons.logging.Log;
2828
import org.apache.commons.logging.LogFactory;
2929

30-
import org.springframework.core.OrderComparator;
30+
import org.springframework.core.annotation.AnnotationAwareOrderComparator;
3131
import org.springframework.core.io.UrlResource;
3232
import org.springframework.util.Assert;
3333
import org.springframework.util.ClassUtils;
@@ -48,20 +48,21 @@
4848
*
4949
* @author Arjen Poutsma
5050
* @author Juergen Hoeller
51+
* @author Sam Brannen
5152
* @since 3.2
5253
*/
5354
public abstract class SpringFactoriesLoader {
5455

5556
/** The location to look for the factories. Can be present in multiple JAR files. */
56-
private static final String FACTORIES_RESOURCE_LOCATION = "META-INF/spring.factories";
57+
public static final String FACTORIES_RESOURCE_LOCATION = "META-INF/spring.factories";
5758

5859
private static final Log logger = LogFactory.getLog(SpringFactoriesLoader.class);
5960

6061

6162
/**
6263
* Load the factory implementations of the given type from the default location,
6364
* using the given class loader.
64-
* <p>The returned factories are ordered in accordance with the {@link OrderComparator}.
65+
* <p>The returned factories are ordered in accordance with the {@link AnnotationAwareOrderComparator}.
6566
* @param factoryClass the interface or abstract class representing the factory
6667
* @param classLoader the ClassLoader to use for loading (can be {@code null} to use the default)
6768
*/
@@ -79,7 +80,7 @@ public static <T> List<T> loadFactories(Class<T> factoryClass, ClassLoader class
7980
for (String factoryName : factoryNames) {
8081
result.add(instantiateFactory(factoryName, factoryClass, classLoaderToUse));
8182
}
82-
OrderComparator.sort(result);
83+
AnnotationAwareOrderComparator.sort(result);
8384
return result;
8485
}
8586

spring-test/src/main/java/org/springframework/test/context/MergedContextConfiguration.java

+13
Original file line numberDiff line numberDiff line change
@@ -79,6 +79,7 @@ public class MergedContextConfiguration implements Serializable {
7979
private final String[] propertySourceProperties;
8080
private final ContextLoader contextLoader;
8181
private final CacheAwareContextLoaderDelegate cacheAwareContextLoaderDelegate;
82+
8283
private final MergedContextConfiguration parent;
8384

8485

@@ -185,6 +186,18 @@ public MergedContextConfiguration(
185186
cacheAwareContextLoaderDelegate, parent);
186187
}
187188

189+
/**
190+
* Create a new {@code MergedContextConfiguration} instance by copying
191+
* all fields from the supplied {@code MergedContextConfiguration}.
192+
* @since 4.1
193+
*/
194+
public MergedContextConfiguration(MergedContextConfiguration mergedConfig) {
195+
this(mergedConfig.testClass, mergedConfig.locations, mergedConfig.classes,
196+
mergedConfig.contextInitializerClasses, mergedConfig.activeProfiles, mergedConfig.propertySourceLocations,
197+
mergedConfig.propertySourceProperties, mergedConfig.contextLoader,
198+
mergedConfig.cacheAwareContextLoaderDelegate, mergedConfig.parent);
199+
}
200+
188201
/**
189202
* Create a new {@code MergedContextConfiguration} instance for the
190203
* supplied parameters.

spring-test/src/main/java/org/springframework/test/context/TestContextBootstrapper.java

+10-2
Original file line numberDiff line numberDiff line change
@@ -63,8 +63,16 @@ public interface TestContextBootstrapper {
6363
* for the test class in the {@link BootstrapContext} associated with this bootstrapper.
6464
* <p>If {@link TestExecutionListeners @TestExecutionListeners} is not
6565
* <em>present</em> on the test class in the {@code BootstrapContext},
66-
* <em>default</em> listeners should be returned. Concrete implementations
67-
* are free to determine what comprises the set of default listeners.
66+
* <em>default</em> listeners should be returned. Furthermore, default
67+
* listeners must be sorted using
68+
* {@link org.springframework.core.annotation.AnnotationAwareOrderComparator
69+
* AnnotationAwareOrderComparator}.
70+
* <p>Concrete implementations are free to determine what comprises the
71+
* set of default listeners. However, by default, the Spring TestContext
72+
* Framework will use the
73+
* {@link org.springframework.core.io.support.SpringFactoriesLoader SpringFactoriesLoader}
74+
* mechanism to look up all {@code TestExecutionListener} class names
75+
* configured in all {@code META-INF/spring.factories} files on the classpath.
6876
* <p>The {@link TestExecutionListeners#inheritListeners() inheritListeners}
6977
* flag of {@link TestExecutionListeners @TestExecutionListeners} must be
7078
* taken into consideration. Specifically, if the {@code inheritListeners}

spring-test/src/main/java/org/springframework/test/context/TestExecutionListener.java

+7-1
Original file line numberDiff line numberDiff line change
@@ -23,7 +23,13 @@
2323
* <p>Concrete implementations must provide a {@code public} no-args constructor,
2424
* so that listeners can be instantiated transparently by tools and configuration
2525
* mechanisms.
26-
* <p>Spring provides the following out-of-the-box implementations:
26+
* <p>Implementations may optionally declare the position in which they should
27+
* be ordered among the chain of default listeners via the
28+
* {@link org.springframework.core.Ordered Order} interface or
29+
* {@link org.springframework.core.annotation.Order @Order} annotation. See
30+
* {@link TestContextBootstrapper#getTestExecutionListeners()} for details.
31+
* <p>Spring provides the following out-of-the-box implementations (all of
32+
* which are annotated with {@code @Order}):
2733
* <ul>
2834
* <li>{@link org.springframework.test.context.web.ServletTestExecutionListener
2935
* ServletTestExecutionListener}</li>

spring-test/src/main/java/org/springframework/test/context/jdbc/SqlScriptsTestExecutionListener.java

+8
Original file line numberDiff line numberDiff line change
@@ -89,6 +89,14 @@ public class SqlScriptsTestExecutionListener extends AbstractTestExecutionListen
8989
private static final Log logger = LogFactory.getLog(SqlScriptsTestExecutionListener.class);
9090

9191

92+
/**
93+
* Returns {@code 5000}.
94+
*/
95+
@Override
96+
public final int getOrder() {
97+
return 5000;
98+
}
99+
92100
/**
93101
* Execute SQL scripts configured via {@link Sql @Sql} for the supplied
94102
* {@link TestContext} <em>before</em> the current test method.

spring-test/src/main/java/org/springframework/test/context/support/AbstractTestContextBootstrapper.java

+62-45
Original file line numberDiff line numberDiff line change
@@ -31,7 +31,9 @@
3131
import org.springframework.context.ApplicationContextInitializer;
3232
import org.springframework.context.ConfigurableApplicationContext;
3333
import org.springframework.core.annotation.AnnotationAttributes;
34+
import org.springframework.core.annotation.AnnotationAwareOrderComparator;
3435
import org.springframework.core.annotation.AnnotationUtils;
36+
import org.springframework.core.io.support.SpringFactoriesLoader;
3537
import org.springframework.test.context.BootstrapContext;
3638
import org.springframework.test.context.CacheAwareContextLoaderDelegate;
3739
import org.springframework.test.context.ContextConfiguration;
@@ -55,11 +57,10 @@
5557
* provides most of the behavior required by a bootstrapper.
5658
*
5759
* <p>Concrete subclasses typically will only need to provide implementations for
58-
* the following {@code abstract} methods:
60+
* the following methods:
5961
* <ul>
60-
* <li>{@link #getDefaultTestExecutionListenerClassNames}
6162
* <li>{@link #getDefaultContextLoaderClass}
62-
* <li>{@link #buildMergedContextConfiguration}
63+
* <li>{@link #processMergedContextConfiguration}
6364
* </ul>
6465
*
6566
* @author Sam Brannen
@@ -98,16 +99,18 @@ public final List<TestExecutionListener> getTestExecutionListeners() {
9899
Class<?> clazz = getBootstrapContext().getTestClass();
99100
Class<TestExecutionListeners> annotationType = TestExecutionListeners.class;
100101
List<Class<? extends TestExecutionListener>> classesList = new ArrayList<Class<? extends TestExecutionListener>>();
102+
boolean usingDefaults = false;
101103

102104
AnnotationDescriptor<TestExecutionListeners> descriptor = MetaAnnotationUtils.findAnnotationDescriptor(clazz,
103105
annotationType);
104106

105107
// Use defaults?
106108
if (descriptor == null) {
107109
if (logger.isDebugEnabled()) {
108-
logger.debug("@TestExecutionListeners is not present for class [" + clazz.getName()
109-
+ "]: using defaults.");
110+
logger.debug(String.format("@TestExecutionListeners is not present for class [%s]: using defaults.",
111+
clazz.getName()));
110112
}
113+
usingDefaults = true;
111114
classesList.addAll(getDefaultTestExecutionListenerClasses());
112115
}
113116
else {
@@ -142,6 +145,20 @@ else if (!ObjectUtils.isEmpty(valueListenerClasses)) {
142145
}
143146
}
144147

148+
List<TestExecutionListener> listeners = instantiateListeners(classesList);
149+
150+
// Sort by Ordered/@Order if we loaded default listeners.
151+
if (usingDefaults) {
152+
AnnotationAwareOrderComparator.sort(listeners);
153+
}
154+
155+
if (logger.isInfoEnabled()) {
156+
logger.info(String.format("Using TestExecutionListeners: %s", listeners));
157+
}
158+
return listeners;
159+
}
160+
161+
private List<TestExecutionListener> instantiateListeners(List<Class<? extends TestExecutionListener>> classesList) {
145162
List<TestExecutionListener> listeners = new ArrayList<TestExecutionListener>(classesList.size());
146163
for (Class<? extends TestExecutionListener> listenerClass : classesList) {
147164
NoClassDefFoundError ncdfe = null;
@@ -194,6 +211,28 @@ protected Set<Class<? extends TestExecutionListener>> getDefaultTestExecutionLis
194211
return defaultListenerClasses;
195212
}
196213

214+
/**
215+
* Get the names of the default {@link TestExecutionListener} classes for
216+
* this bootstrapper.
217+
* <p>The default implementation looks up all
218+
* {@code org.springframework.test.context.TestExecutionListener} entries
219+
* configured in all {@code META-INF/spring.factories} files on the classpath.
220+
* <p>This method is invoked by {@link #getDefaultTestExecutionListenerClasses()}.
221+
* @return an <em>unmodifiable</em> list of names of default {@code TestExecutionListener}
222+
* classes
223+
* @see SpringFactoriesLoader#loadFactoryNames
224+
*/
225+
protected List<String> getDefaultTestExecutionListenerClassNames() {
226+
final List<String> classNames = SpringFactoriesLoader.loadFactoryNames(TestExecutionListener.class,
227+
getClass().getClassLoader());
228+
229+
if (logger.isInfoEnabled()) {
230+
logger.info(String.format("Loaded default TestExecutionListener class names from location [%s]: %s",
231+
SpringFactoriesLoader.FACTORIES_RESOURCE_LOCATION, classNames));
232+
}
233+
return Collections.unmodifiableList(classNames);
234+
}
235+
197236
/**
198237
* {@inheritDoc}
199238
*/
@@ -302,9 +341,11 @@ private MergedContextConfiguration buildMergedContextConfiguration(Class<?> test
302341
String[] activeProfiles = ActiveProfilesUtils.resolveActiveProfiles(testClass);
303342
MergedTestPropertySources mergedTestPropertySources = TestPropertySourceUtils.buildMergedTestPropertySources(testClass);
304343

305-
return buildMergedContextConfiguration(testClass, locations, classes, initializerClasses, activeProfiles,
306-
mergedTestPropertySources.getLocations(), mergedTestPropertySources.getProperties(), contextLoader,
307-
cacheAwareContextLoaderDelegate, parentConfig);
344+
MergedContextConfiguration mergedConfig = new MergedContextConfiguration(testClass, locations, classes,
345+
initializerClasses, activeProfiles, mergedTestPropertySources.getLocations(),
346+
mergedTestPropertySources.getProperties(), contextLoader, cacheAwareContextLoaderDelegate, parentConfig);
347+
348+
return processMergedContextConfiguration(mergedConfig);
308349
}
309350

310351
/**
@@ -383,15 +424,6 @@ private Class<? extends ContextLoader> resolveExplicitContextLoaderClass(
383424
return null;
384425
}
385426

386-
/**
387-
* Get the names of the default {@link TestExecutionListener} classes for
388-
* this bootstrapper.
389-
* <p>This method is invoked by {@link #getDefaultTestExecutionListenerClasses()}.
390-
* @return an <em>unmodifiable</em> list of names of default {@code
391-
* TestExecutionListener} classes
392-
*/
393-
protected abstract List<String> getDefaultTestExecutionListenerClassNames();
394-
395427
/**
396428
* Determine the default {@link ContextLoader} class to use for the supplied
397429
* test class.
@@ -403,34 +435,19 @@ private Class<? extends ContextLoader> resolveExplicitContextLoaderClass(
403435
protected abstract Class<? extends ContextLoader> getDefaultContextLoaderClass(Class<?> testClass);
404436

405437
/**
406-
* Build a {@link MergedContextConfiguration} instance from the supplied,
407-
* merged values.
408-
* <p>Concrete subclasses typically will only need to instantiate
409-
* {@link MergedContextConfiguration} (or a specialized subclass thereof)
410-
* from the provided values; further processing and merging of values is likely
411-
* unnecessary.
412-
* @param testClass the test class for which the {@code MergedContextConfiguration}
413-
* should be built (must not be {@code null})
414-
* @param locations the merged resource locations
415-
* @param classes the merged annotated classes
416-
* @param initializerClasses the merged context initializer classes
417-
* @param activeProfiles the merged active bean definition profiles
418-
* @param propertySourceLocations the merged {@code PropertySource} locations
419-
* @param propertySourceProperties the merged {@code PropertySource} properties
420-
* @param contextLoader the resolved {@code ContextLoader}
421-
* @param cacheAwareContextLoaderDelegate the cache-aware context loader delegate
422-
* to be provided to the instantiated {@code MergedContextConfiguration}
423-
* @param parentConfig the merged context configuration for the parent application
424-
* context in a context hierarchy, or {@code null} if there is no parent
425-
* @return the fully initialized {@code MergedContextConfiguration}
438+
* Process the supplied, newly instantiated {@link MergedContextConfiguration} instance.
439+
* <p>The returned {@link MergedContextConfiguration} instance may be a wrapper
440+
* around or a replacement for the original.
441+
* <p>The default implementation simply returns the supplied instance unmodified.
442+
* <p>Concrete subclasses may choose to return a specialized subclass of
443+
* {@link MergedContextConfiguration} based on properties in the supplied instance.
444+
* @param mergedConfig the {@code MergedContextConfiguration} to process;
445+
* never {@code null}
446+
* @return a fully initialized {@code MergedContextConfiguration}; never
447+
* {@code null}
426448
*/
427-
protected abstract MergedContextConfiguration buildMergedContextConfiguration(
428-
Class<?> testClass,
429-
String[] locations,
430-
Class<?>[] classes,
431-
Set<Class<? extends ApplicationContextInitializer<? extends ConfigurableApplicationContext>>> initializerClasses,
432-
String[] activeProfiles, String[] propertySourceLocations, String[] propertySourceProperties,
433-
ContextLoader contextLoader, CacheAwareContextLoaderDelegate cacheAwareContextLoaderDelegate,
434-
MergedContextConfiguration parentConfig);
449+
protected MergedContextConfiguration processMergedContextConfiguration(MergedContextConfiguration mergedConfig) {
450+
return mergedConfig;
451+
}
435452

436453
}

spring-test/src/main/java/org/springframework/test/context/support/AbstractTestExecutionListener.java

+15-2
Original file line numberDiff line numberDiff line change
@@ -1,5 +1,5 @@
11
/*
2-
* Copyright 2002-2012 the original author or authors.
2+
* Copyright 2002-2014 the original author or authors.
33
*
44
* Licensed under the Apache License, Version 2.0 (the "License");
55
* you may not use this file except in compliance with the License.
@@ -16,6 +16,7 @@
1616

1717
package org.springframework.test.context.support;
1818

19+
import org.springframework.core.Ordered;
1920
import org.springframework.test.context.TestContext;
2021
import org.springframework.test.context.TestExecutionListener;
2122

@@ -28,7 +29,7 @@
2829
* @author Juergen Hoeller
2930
* @since 2.5
3031
*/
31-
public abstract class AbstractTestExecutionListener implements TestExecutionListener {
32+
public abstract class AbstractTestExecutionListener implements TestExecutionListener, Ordered {
3233

3334
/**
3435
* The default implementation is <em>empty</em>. Can be overridden by
@@ -75,4 +76,16 @@ public void afterTestClass(TestContext testContext) throws Exception {
7576
/* no-op */
7677
}
7778

79+
/**
80+
* The default implementation returns {@link Ordered#LOWEST_PRECEDENCE},
81+
* thereby ensuring that custom listeners are ordered after default
82+
* listeners supplied by the framework. Can be overridden by subclasses
83+
* as necessary.
84+
* @since 4.1
85+
*/
86+
@Override
87+
public int getOrder() {
88+
return Ordered.LOWEST_PRECEDENCE;
89+
}
90+
7891
}

0 commit comments

Comments
 (0)