My Spring Boot应用程序启动如下:
new SpringApplicationBuilder()
.sources(ParentCtxConfig.class)
.child(ChildFirstCtxConfig.class)
.sibling(ChildSecondCtxConfig.class)
.run(args);
配置类使用@SpringBootApplication
进行注释。因此,我有一个根上下文和两个子Web上下文。
我想编写集成测试,我希望在那里有相同的上下文层次结构。我希望至少用他的父上下文(ChildFirstCtxConfig.class
)测试第一个子上下文(用ParentCtxConfig.class
配置)。我怎样才能做到这一点?
目前我在测试中自动ApplicationContext
,所以我可以检查它。我在测试中有这个类注释:
@RunWith(SpringRunner.class)
@SpringBootTest(classes = { ParentCtxConfig.class, ChildFirstCtxConfig.class }, webEnvironment = WebEnvironment.RANDOM_PORT)
但这将产生单个上下文,我想要父子层次结构。
我假设我应该使用@ContextHierarchy
注释来注释我的测试。
将我的测试注释更改为与之前的示例完全相同:
@RunWith(SpringRunner.class)
@ContextConfiguration(classes = { ParentCtxConfig.class, ChildFirstCtxConfig.class })
@SpringBootTest(webEnvironment = WebEnvironment.RANDOM_PORT)
但是如果我想介绍@ContextHierarchy
并且有类似的东西:
@RunWith(SpringRunner.class)
@ContextHierarchy({
@ContextConfiguration(name = "root", classes = ParentCtxConfig.class),
@ContextConfiguration(name = "child", classes = ChildFirstCtxConfig.class)
})
@SpringBootTest(webEnvironment = WebEnvironment.RANDOM_PORT)
由于在父上下文中定义的bean无法在子上下文中找到/自动装配,因此未启动上下文。设置loader = SpringBootContextLoader.class
无济于事。
示例代码:GitHub
答案 0 :(得分:6)
更新 This issue was fixed in Spring Boot 1.5.0。
这是SpringBootContextLoader
的限制。准确移动,这是ContextCustomizer
的限制。您可以使用配置父上下文的自定义上下文加载器或spring.factories
需要在org.springframework.test.context.ContextCustomizerFactory=\
com.alex.demo.ctx.HierarchyContextCustomizerFactory
中列出的工厂来解决此问题。以下是后者的一个粗略示例:
package com.alex.demo.ctx;
import java.util.List;
import org.springframework.context.ConfigurableApplicationContext;
import org.springframework.test.context.ContextConfigurationAttributes;
import org.springframework.test.context.ContextCustomizer;
import org.springframework.test.context.ContextCustomizerFactory;
import org.springframework.test.context.MergedContextConfiguration;
public class HierarchyContextCustomizerFactory implements ContextCustomizerFactory {
@Override
public ContextCustomizer createContextCustomizer(Class<?> testClass,
List<ContextConfigurationAttributes> configAttributes) {
return new ContextCustomizer() {
@Override
public void customizeContext(ConfigurableApplicationContext context,
MergedContextConfiguration mergedConfig) {
if (mergedConfig.getParentApplicationContext() != null) {
context.setParent(mergedConfig.getParentApplicationContext());
}
}
};
}
}
angular.element(dirElementInput).val('Some text').trigger('input');