AbstractContextLoader.java 14.3 KB
Newer Older
A
Arjen Poutsma 已提交
1
/*
2
 * Copyright 2002-2016 the original author or authors.
A
Arjen Poutsma 已提交
3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18
 *
 * Licensed under the Apache License, Version 2.0 (the "License");
 * you may not use this file except in compliance with the License.
 * You may obtain a copy of the License at
 *
 *      http://www.apache.org/licenses/LICENSE-2.0
 *
 * Unless required by applicable law or agreed to in writing, software
 * distributed under the License is distributed on an "AS IS" BASIS,
 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
 * See the License for the specific language governing permissions and
 * limitations under the License.
 */

package org.springframework.test.context.support;

19 20 21 22
import java.util.ArrayList;
import java.util.List;
import java.util.Set;

23 24
import org.apache.commons.logging.Log;
import org.apache.commons.logging.LogFactory;
25

26
import org.springframework.beans.BeanUtils;
A
Arjen Poutsma 已提交
27
import org.springframework.context.ApplicationContext;
28
import org.springframework.context.ApplicationContextException;
29 30 31 32
import org.springframework.context.ApplicationContextInitializer;
import org.springframework.context.ConfigurableApplicationContext;
import org.springframework.core.GenericTypeResolver;
import org.springframework.core.annotation.AnnotationAwareOrderComparator;
33
import org.springframework.core.env.PropertySource;
34
import org.springframework.core.io.ClassPathResource;
35
import org.springframework.test.context.ContextConfigurationAttributes;
A
Arjen Poutsma 已提交
36
import org.springframework.test.context.ContextLoader;
37
import org.springframework.test.context.MergedContextConfiguration;
38
import org.springframework.test.context.SmartContextLoader;
39
import org.springframework.test.context.util.TestContextResourceUtils;
A
Arjen Poutsma 已提交
40 41 42 43 44 45
import org.springframework.util.Assert;
import org.springframework.util.ClassUtils;
import org.springframework.util.ObjectUtils;
import org.springframework.util.ResourceUtils;

/**
46
 * Abstract application context loader that provides a basis for all concrete
47
 * implementations of the {@link ContextLoader} SPI. Provides a
J
Juergen Hoeller 已提交
48
 * <em>Template Method</em> based approach for {@link #processLocations processing}
49
 * resource locations.
S
Stevo Slavic 已提交
50
 *
51
 * <p>As of Spring 3.1, {@code AbstractContextLoader} also provides a basis
52
 * for all concrete implementations of the {@link SmartContextLoader} SPI. For
S
Stevo Slavic 已提交
53 54 55
 * backwards compatibility with the {@code ContextLoader} SPI,
 * {@link #processContextConfiguration(ContextConfigurationAttributes)} delegates
 * to {@link #processLocations(Class, String...)}.
A
Arjen Poutsma 已提交
56 57 58 59 60
 *
 * @author Sam Brannen
 * @author Juergen Hoeller
 * @since 2.5
 * @see #generateDefaultLocations
S
Sam Brannen 已提交
61
 * @see #getResourceSuffixes
A
Arjen Poutsma 已提交
62 63
 * @see #modifyLocations
 */
64 65
public abstract class AbstractContextLoader implements SmartContextLoader {

66
	private static final String[] EMPTY_STRING_ARRAY = new String[0];
J
Juergen Hoeller 已提交
67 68 69

	private static final Log logger = LogFactory.getLog(AbstractContextLoader.class);

70

71 72
	// --- SmartContextLoader -----------------------------------------------

73
	/**
74
	 * For backwards compatibility with the {@link ContextLoader} SPI, the
S
Stevo Slavic 已提交
75
	 * default implementation simply delegates to {@link #processLocations(Class, String...)},
76 77 78 79 80 81 82 83
	 * passing it the {@link ContextConfigurationAttributes#getDeclaringClass()
	 * declaring class} and {@link ContextConfigurationAttributes#getLocations()
	 * resource locations} retrieved from the supplied
	 * {@link ContextConfigurationAttributes configuration attributes}. The
	 * processed locations are then
	 * {@link ContextConfigurationAttributes#setLocations(String[]) set} in
	 * the supplied configuration attributes.
	 * <p>Can be overridden in subclasses &mdash; for example, to process
84
	 * annotated classes instead of resource locations.
85
	 * @since 3.1
S
Stevo Slavic 已提交
86
	 * @see #processLocations(Class, String...)
87
	 */
88
	@Override
89
	public void processContextConfiguration(ContextConfigurationAttributes configAttributes) {
90 91
		String[] processedLocations =
				processLocations(configAttributes.getDeclaringClass(), configAttributes.getLocations());
92 93
		configAttributes.setLocations(processedLocations);
	}
A
Arjen Poutsma 已提交
94

95
	/**
96
	 * Prepare the {@link ConfigurableApplicationContext} created by this
97 98 99 100
	 * {@code SmartContextLoader} <i>before</i> bean definitions are read.
	 * <p>The default implementation:
	 * <ul>
	 * <li>Sets the <em>active bean definition profiles</em> from the supplied
101
	 * {@code MergedContextConfiguration} in the
102 103 104 105 106 107 108 109
	 * {@link org.springframework.core.env.Environment Environment} of the
	 * context.</li>
	 * <li>Adds {@link PropertySource PropertySources} for all
	 * {@linkplain MergedContextConfiguration#getPropertySourceLocations()
	 * resource locations} and
	 * {@linkplain MergedContextConfiguration#getPropertySourceProperties()
	 * inlined properties} from the supplied {@code MergedContextConfiguration}
	 * to the {@code Environment} of the context.</li>
110
	 * <li>Determines what (if any) context initializer classes have been supplied
111 112
	 * via the {@code MergedContextConfiguration} and instantiates and
	 * {@linkplain ApplicationContextInitializer#initialize invokes} each with the
113
	 * given application context.</li>
114 115 116 117 118
	 * <ul>
	 * <li>Any {@code ApplicationContextInitializers} implementing
	 * {@link org.springframework.core.Ordered Ordered} or annotated with {@link
	 * org.springframework.core.annotation.Order @Order} will be sorted appropriately.</li>
	 * </ul>
119 120 121
	 * </ul>
	 * @param context the newly created application context
	 * @param mergedConfig the merged context configuration
J
Juergen Hoeller 已提交
122
	 * @since 3.2
123 124
	 * @see TestPropertySourceUtils#addPropertiesFilesToEnvironment
	 * @see TestPropertySourceUtils#addInlinedPropertiesToEnvironment
125 126 127 128 129 130
	 * @see ApplicationContextInitializer#initialize(ConfigurableApplicationContext)
	 * @see #loadContext(MergedContextConfiguration)
	 * @see ConfigurableApplicationContext#setId
	 */
	protected void prepareContext(ConfigurableApplicationContext context, MergedContextConfiguration mergedConfig) {
		context.getEnvironment().setActiveProfiles(mergedConfig.getActiveProfiles());
131
		TestPropertySourceUtils.addPropertiesFilesToEnvironment(context, mergedConfig.getPropertySourceLocations());
132
		TestPropertySourceUtils.addInlinedPropertiesToEnvironment(context, mergedConfig.getPropertySourceProperties());
133 134
		invokeApplicationContextInitializers(context, mergedConfig);
	}
135

136 137 138
	@SuppressWarnings("unchecked")
	private void invokeApplicationContextInitializers(ConfigurableApplicationContext context,
			MergedContextConfiguration mergedConfig) {
139 140 141

		Set<Class<? extends ApplicationContextInitializer<? extends ConfigurableApplicationContext>>> initializerClasses =
				mergedConfig.getContextInitializerClasses();
J
Juergen Hoeller 已提交
142
		if (initializerClasses.isEmpty()) {
143 144 145 146
			// no ApplicationContextInitializers have been declared -> nothing to do
			return;
		}

147
		List<ApplicationContextInitializer<ConfigurableApplicationContext>> initializerInstances = new ArrayList<ApplicationContextInitializer<ConfigurableApplicationContext>>();
J
Juergen Hoeller 已提交
148
		Class<?> contextClass = context.getClass();
149 150

		for (Class<? extends ApplicationContextInitializer<? extends ConfigurableApplicationContext>> initializerClass : initializerClasses) {
151 152 153 154 155 156 157 158 159
			Class<?> initializerContextClass =
					GenericTypeResolver.resolveTypeArgument(initializerClass, ApplicationContextInitializer.class);
			if (initializerContextClass != null && !initializerContextClass.isInstance(context)) {
				throw new ApplicationContextException(String.format(
						"Could not apply context initializer [%s] since its generic parameter [%s] " +
						"is not assignable from the type of application context used by this " +
						"context loader: [%s]", initializerClass.getName(), initializerContextClass.getName(),
						contextClass.getName()));
			}
160 161 162
			initializerInstances.add((ApplicationContextInitializer<ConfigurableApplicationContext>) BeanUtils.instantiateClass(initializerClass));
		}

J
Juergen Hoeller 已提交
163
		AnnotationAwareOrderComparator.sort(initializerInstances);
164 165 166 167 168
		for (ApplicationContextInitializer<ConfigurableApplicationContext> initializer : initializerInstances) {
			initializer.initialize(context);
		}
	}

169

170 171
	// --- ContextLoader -------------------------------------------------------

A
Arjen Poutsma 已提交
172
	/**
S
Sam Brannen 已提交
173 174 175 176 177 178 179
	 * If the supplied {@code locations} are {@code null} or <em>empty</em>
	 * and {@link #isGenerateDefaultLocations()} returns {@code true},
	 * default locations will be {@link #generateDefaultLocations(Class)
	 * generated} (i.e., detected) for the specified {@link Class class}
	 * and the configured {@linkplain #getResourceSuffixes() resource suffixes};
	 * otherwise, the supplied {@code locations} will be
	 * {@linkplain #modifyLocations modified} if necessary and returned.
A
Arjen Poutsma 已提交
180 181 182
	 * @param clazz the class with which the locations are associated: to be
	 * used when generating default locations
	 * @param locations the unmodified locations to use for loading the
183
	 * application context (can be {@code null} or empty)
184
	 * @return a processed array of application context resource locations
185
	 * @since 2.5
186
	 * @see #isGenerateDefaultLocations()
S
Stevo Slavic 已提交
187 188 189 190
	 * @see #generateDefaultLocations(Class)
	 * @see #modifyLocations(Class, String...)
	 * @see org.springframework.test.context.ContextLoader#processLocations(Class, String...)
	 * @see #processContextConfiguration(ContextConfigurationAttributes)
A
Arjen Poutsma 已提交
191
	 */
192
	@Override
A
Arjen Poutsma 已提交
193
	public final String[] processLocations(Class<?> clazz, String... locations) {
194 195
		return (ObjectUtils.isEmpty(locations) && isGenerateDefaultLocations()) ?
				generateDefaultLocations(clazz) : modifyLocations(clazz, locations);
A
Arjen Poutsma 已提交
196 197 198
	}

	/**
199
	 * Generate the default classpath resource locations array based on the
A
Arjen Poutsma 已提交
200
	 * supplied class.
201
	 * <p>For example, if the supplied class is {@code com.example.MyTest},
A
Arjen Poutsma 已提交
202
	 * the generated locations will contain a single string with a value of
S
Sam Brannen 已提交
203 204 205 206
	 * {@code "classpath:com/example/MyTest<suffix>"}, where {@code <suffix>}
	 * is the value of the first configured
	 * {@linkplain #getResourceSuffixes() resource suffix} for which the
	 * generated location actually exists in the classpath.
207
	 * <p>As of Spring 3.1, the implementation of this method adheres to the
S
Stevo Slavic 已提交
208
	 * contract defined in the {@link SmartContextLoader} SPI. Specifically,
209 210 211
	 * this method will <em>preemptively</em> verify that the generated default
	 * location actually exists. If it does not exist, this method will log a
	 * warning and return an empty array.
A
Arjen Poutsma 已提交
212 213 214 215
	 * <p>Subclasses can override this method to implement a different
	 * <em>default location generation</em> strategy.
	 * @param clazz the class for which the default locations are to be generated
	 * @return an array of default application context resource locations
216
	 * @since 2.5
S
Sam Brannen 已提交
217
	 * @see #getResourceSuffixes()
A
Arjen Poutsma 已提交
218 219 220
	 */
	protected String[] generateDefaultLocations(Class<?> clazz) {
		Assert.notNull(clazz, "Class must not be null");
221

S
Sam Brannen 已提交
222 223 224 225 226 227 228 229 230
		String[] suffixes = getResourceSuffixes();
		for (String suffix : suffixes) {
			Assert.hasText(suffix, "Resource suffix must not be empty");
			String resourcePath = ClassUtils.convertClassNameToResourcePath(clazz.getName()) + suffix;
			String prefixedResourcePath = ResourceUtils.CLASSPATH_URL_PREFIX + resourcePath;
			ClassPathResource classPathResource = new ClassPathResource(resourcePath);
			if (classPathResource.exists()) {
				if (logger.isInfoEnabled()) {
					logger.info(String.format("Detected default resource location \"%s\" for test class [%s]",
231
							prefixedResourcePath, clazz.getName()));
S
Sam Brannen 已提交
232
				}
233
				return new String[] {prefixedResourcePath};
234
			}
S
Sam Brannen 已提交
235
			else if (logger.isDebugEnabled()) {
236 237
				logger.debug(String.format("Did not detect default resource location for test class [%s]: " +
						"%s does not exist", clazz.getName(), classPathResource));
J
Juergen Hoeller 已提交
238
			}
239
		}
S
Sam Brannen 已提交
240 241

		if (logger.isInfoEnabled()) {
242 243
			logger.info(String.format("Could not detect default resource locations for test class [%s]: " +
					"no resource found for suffixes %s.", clazz.getName(), ObjectUtils.nullSafeToString(suffixes)));
S
Sam Brannen 已提交
244 245 246
		}

		return EMPTY_STRING_ARRAY;
A
Arjen Poutsma 已提交
247 248 249
	}

	/**
S
Sam Brannen 已提交
250
	 * Generate a modified version of the supplied locations array and return it.
251 252
	 * <p>The default implementation simply delegates to
	 * {@link TestContextResourceUtils#convertToClasspathResourcePaths}.
A
Arjen Poutsma 已提交
253 254 255 256 257
	 * <p>Subclasses can override this method to implement a different
	 * <em>location modification</em> strategy.
	 * @param clazz the class with which the locations are associated
	 * @param locations the resource locations to be modified
	 * @return an array of modified application context resource locations
258
	 * @since 2.5
A
Arjen Poutsma 已提交
259 260
	 */
	protected String[] modifyLocations(Class<?> clazz, String... locations) {
261
		return TestContextResourceUtils.convertToClasspathResourcePaths(clazz, locations);
A
Arjen Poutsma 已提交
262 263 264 265
	}

	/**
	 * Determine whether or not <em>default</em> resource locations should be
266 267
	 * generated if the {@code locations} provided to
	 * {@link #processLocations(Class, String...)} are {@code null} or empty.
268 269
	 * <p>As of Spring 3.1, the semantics of this method have been overloaded
	 * to include detection of either default resource locations or default
S
Stevo Slavic 已提交
270
	 * configuration classes. Consequently, this method can also be used to
271
	 * determine whether or not <em>default</em> configuration classes should be
272
	 * detected if the {@code classes} present in the
273
	 * {@link ContextConfigurationAttributes configuration attributes} supplied
S
Stevo Slavic 已提交
274
	 * to {@link #processContextConfiguration(ContextConfigurationAttributes)}
275
	 * are {@code null} or empty.
A
Arjen Poutsma 已提交
276
	 * <p>Can be overridden by subclasses to change the default behavior.
277
	 * @return always {@code true} by default
278
	 * @since 2.5
A
Arjen Poutsma 已提交
279 280 281 282 283 284
	 */
	protected boolean isGenerateDefaultLocations() {
		return true;
	}

	/**
285 286
	 * Get the suffixes to append to {@link ApplicationContext} resource locations
	 * when detecting default locations.
S
Sam Brannen 已提交
287 288 289 290 291 292 293 294
	 * <p>The default implementation simply wraps the value returned by
	 * {@link #getResourceSuffix()} in a single-element array, but this
	 * can be overridden by subclasses in order to support multiple suffixes.
	 * @return the resource suffixes; never {@code null} or empty
	 * @since 4.1
	 * @see #generateDefaultLocations(Class)
	 */
	protected String[] getResourceSuffixes() {
295
		return new String[] {getResourceSuffix()};
S
Sam Brannen 已提交
296 297
	}

298 299 300 301 302 303 304 305 306 307 308 309 310 311
	/**
	 * Get the suffix to append to {@link ApplicationContext} resource locations
	 * when detecting default locations.
	 * <p>Subclasses must provide an implementation of this method that returns
	 * a single suffix. Alternatively subclasses may provide a  <em>no-op</em>
	 * implementation of this method and override {@link #getResourceSuffixes()}
	 * in order to provide multiple custom suffixes.
	 * @return the resource suffix; never {@code null} or empty
	 * @since 2.5
	 * @see #generateDefaultLocations(Class)
	 * @see #getResourceSuffixes()
	 */
	protected abstract String getResourceSuffix();

A
Arjen Poutsma 已提交
312
}