org.springframework.test.context.support.DefaultTestContext Maven / Gradle / Ivy
Show all versions of spring-test Show documentation
/*
* Copyright 2002-2023 the original author or authors.
*
* 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
*
* https://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;
import java.lang.reflect.Method;
import java.util.Map;
import java.util.concurrent.ConcurrentHashMap;
import java.util.function.Function;
import org.springframework.context.ApplicationContext;
import org.springframework.context.ConfigurableApplicationContext;
import org.springframework.core.style.DefaultToStringStyler;
import org.springframework.core.style.SimpleValueStyler;
import org.springframework.core.style.ToStringCreator;
import org.springframework.lang.Nullable;
import org.springframework.test.annotation.DirtiesContext.HierarchyMode;
import org.springframework.test.context.CacheAwareContextLoaderDelegate;
import org.springframework.test.context.MergedContextConfiguration;
import org.springframework.test.context.MethodInvoker;
import org.springframework.test.context.TestContext;
import org.springframework.util.Assert;
import org.springframework.util.StringUtils;
/**
* Default implementation of the {@link TestContext} interface.
*
* @author Sam Brannen
* @author Juergen Hoeller
* @author Rob Harrop
* @since 4.0
*/
@SuppressWarnings("serial")
public class DefaultTestContext implements TestContext {
private static final long serialVersionUID = -5827157174866681233L;
private final Map attributes = new ConcurrentHashMap<>(4);
private final CacheAwareContextLoaderDelegate cacheAwareContextLoaderDelegate;
private final MergedContextConfiguration mergedConfig;
private final Class testClass;
@Nullable
private volatile Object testInstance;
@Nullable
private volatile Method testMethod;
@Nullable
private volatile Throwable testException;
private volatile MethodInvoker methodInvoker = MethodInvoker.DEFAULT_INVOKER;
/**
* Copy constructor for creating a new {@code DefaultTestContext}
* based on the attributes and immutable state of the supplied context.
* Immutable state includes all arguments supplied to the
* {@linkplain #DefaultTestContext(Class, MergedContextConfiguration,
* CacheAwareContextLoaderDelegate) standard constructor}.
* @throws NullPointerException if the supplied {@code DefaultTestContext}
* is {@code null}
*/
public DefaultTestContext(DefaultTestContext testContext) {
this(testContext.testClass, testContext.mergedConfig,
testContext.cacheAwareContextLoaderDelegate);
this.attributes.putAll(testContext.attributes);
}
/**
* Construct a new {@code DefaultTestContext} from the supplied arguments.
* @param testClass the test class for this test context
* @param mergedConfig the merged application context
* configuration for this test context
* @param cacheAwareContextLoaderDelegate the delegate to use for loading
* and closing the application context for this test context
*/
public DefaultTestContext(Class testClass, MergedContextConfiguration mergedConfig,
CacheAwareContextLoaderDelegate cacheAwareContextLoaderDelegate) {
Assert.notNull(testClass, "Test Class must not be null");
Assert.notNull(mergedConfig, "MergedContextConfiguration must not be null");
Assert.notNull(cacheAwareContextLoaderDelegate, "CacheAwareContextLoaderDelegate must not be null");
this.testClass = testClass;
this.mergedConfig = mergedConfig;
this.cacheAwareContextLoaderDelegate = cacheAwareContextLoaderDelegate;
}
/**
* Determine if the {@linkplain ApplicationContext application context} for
* this test context is present in the context cache.
* @return {@code true} if the application context has already been loaded
* and stored in the context cache
* @since 5.2
* @see #getApplicationContext()
* @see CacheAwareContextLoaderDelegate#isContextLoaded
*/
@Override
public boolean hasApplicationContext() {
return this.cacheAwareContextLoaderDelegate.isContextLoaded(this.mergedConfig);
}
/**
* Get the {@linkplain ApplicationContext application context} for this
* test context.
*
The default implementation delegates to the {@link CacheAwareContextLoaderDelegate}
* that was supplied when this {@code TestContext} was constructed.
* @throws IllegalStateException if the context returned by the context
* loader delegate is not active (i.e., has been closed)
* @see CacheAwareContextLoaderDelegate#loadContext
*/
@Override
public ApplicationContext getApplicationContext() {
ApplicationContext context = this.cacheAwareContextLoaderDelegate.loadContext(this.mergedConfig);
if (context instanceof ConfigurableApplicationContext cac) {
Assert.state(cac.isActive(), () -> """
The ApplicationContext loaded for %s is not active. \
This may be due to one of the following reasons: \
1) the context was closed programmatically by user code; \
2) the context was closed during parallel test execution either \
according to @DirtiesContext semantics or due to automatic eviction \
from the ContextCache due to a maximum cache size policy."""
.formatted(this.mergedConfig));
}
return context;
}
/**
* Mark the {@linkplain ApplicationContext application context} associated
* with this test context as dirty (i.e., by removing it from the
* context cache and closing it).
*
The default implementation delegates to the {@link CacheAwareContextLoaderDelegate}
* that was supplied when this {@code TestContext} was constructed.
* @see CacheAwareContextLoaderDelegate#closeContext
*/
@Override
public void markApplicationContextDirty(@Nullable HierarchyMode hierarchyMode) {
this.cacheAwareContextLoaderDelegate.closeContext(this.mergedConfig, hierarchyMode);
}
@Override
public final Class getTestClass() {
return this.testClass;
}
@Override
public final Object getTestInstance() {
Object testInstance = this.testInstance;
Assert.state(testInstance != null, "No test instance");
return testInstance;
}
@Override
public final Method getTestMethod() {
Method testMethod = this.testMethod;
Assert.state(testMethod != null, "No test method");
return testMethod;
}
@Override
@Nullable
public final Throwable getTestException() {
return this.testException;
}
@Override
public void updateState(@Nullable Object testInstance, @Nullable Method testMethod, @Nullable Throwable testException) {
this.testInstance = testInstance;
this.testMethod = testMethod;
this.testException = testException;
}
@Override
public final void setMethodInvoker(MethodInvoker methodInvoker) {
Assert.notNull(methodInvoker, "MethodInvoker must not be null");
this.methodInvoker = methodInvoker;
}
@Override
public final MethodInvoker getMethodInvoker() {
return this.methodInvoker;
}
@Override
public void setAttribute(String name, @Nullable Object value) {
Assert.notNull(name, "Name must not be null");
synchronized (this.attributes) {
if (value != null) {
this.attributes.put(name, value);
}
else {
this.attributes.remove(name);
}
}
}
@Override
@Nullable
public Object getAttribute(String name) {
Assert.notNull(name, "Name must not be null");
return this.attributes.get(name);
}
@Override
@SuppressWarnings("unchecked")
public T computeAttribute(String name, Function computeFunction) {
Assert.notNull(name, "Name must not be null");
Assert.notNull(computeFunction, "Compute function must not be null");
Object value = this.attributes.computeIfAbsent(name, computeFunction);
Assert.state(value != null,
() -> "Compute function must not return null for attribute named '%s'".formatted(name));
return (T) value;
}
@Override
@Nullable
public Object removeAttribute(String name) {
Assert.notNull(name, "Name must not be null");
return this.attributes.remove(name);
}
@Override
public boolean hasAttribute(String name) {
Assert.notNull(name, "Name must not be null");
return this.attributes.containsKey(name);
}
@Override
public String[] attributeNames() {
synchronized (this.attributes) {
return StringUtils.toStringArray(this.attributes.keySet());
}
}
/**
* Provide a String representation of this test context's state.
*/
@Override
public String toString() {
return new ToStringCreator(this, new DefaultToStringStyler(new SimpleValueStyler()))
.append("testClass", this.testClass)
.append("testInstance", this.testInstance)
.append("testMethod", this.testMethod)
.append("testException", this.testException)
.append("mergedContextConfiguration", this.mergedConfig)
.append("attributes", this.attributes)
.toString();
}
}