Spring 在 Atlassian Bitbucket 管道上使用 @WebMvcTest 启动测试失败
Spring Boot Test with @WebMvcTest fails on Atlassian Bitbucket Pipeline
当 运行在本地 @WebMvcTest
时,我没有问题(Spring Boot 1.5.8,gradle 4.6):
@RunWith(SpringRunner.class)
@WebMvcTest(VZNFCController.class)
public class VZNFCControllerTest {
private VZNFCTagAction action1 = new VZNFCTagAction();
private VZNFCTagAction action2 = new VZNFCTagAction();
@Before
public void setUp(){
action1.setMessage("message1");
action2.setMessage("message2");
}
@Autowired
private MockMvc mvc;
@MockBean
private VZNFCTagActionRepository actionRepository;
@MockBean
private MappingMongoConverter mongoConverter;
@Test
@WithMockUser
public void testGetTagList() throws Exception {
given(this.actionRepository.findAll())
.willReturn(Arrays.asList(action1, action2));
this.mvc.perform(get("/api/nfc/tags")
.accept(MediaType.APPLICATION_JSON_UTF8))
.andExpect(status().isOk());
}
}
但是,当我上传到 Atlassian Bitbucket 并在那里 运行 ./gradlew test --stacktrace
时,我得到以下信息:
java.lang.IllegalStateException: Failed to load ApplicationContext
at org.springframework.test.context.cache.DefaultCacheAwareContextLoaderDelegate.loadContext(DefaultCacheAwareContextLoaderDelegate.java:124)
at org.springframework.test.context.support.DefaultTestContext.getApplicationContext(DefaultTestContext.java:83)
at org.springframework.test.context.support.DependencyInjectionTestExecutionListener.injectDependencies(DependencyInjectionTestExecutionListener.java:117)
at org.springframework.test.context.support.DependencyInjectionTestExecutionListener.prepareTestInstance(DependencyInjectionTestExecutionListener.java:83)
at org.springframework.boot.test.autoconfigure.SpringBootDependencyInjectionTestExecutionListener.prepareTestInstance(SpringBootDependencyInjectionTestExecutionListener.java:44)
at org.springframework.test.context.TestContextManager.prepareTestInstance(TestContextManager.java:230)
at org.springframework.test.context.junit4.SpringJUnit4ClassRunner.createTest(SpringJUnit4ClassRunner.java:228)
at org.springframework.test.context.junit4.SpringJUnit4ClassRunner.runReflectiveCall(SpringJUnit4ClassRunner.java:287)
at org.junit.internal.runners.model.ReflectiveCallable.run(ReflectiveCallable.java:12)
at org.springframework.test.context.junit4.SpringJUnit4ClassRunner.methodBlock(SpringJUnit4ClassRunner.java:289)
at org.springframework.test.context.junit4.SpringJUnit4ClassRunner.runChild(SpringJUnit4ClassRunner.java:247)
at org.springframework.test.context.junit4.SpringJUnit4ClassRunner.runChild(SpringJUnit4ClassRunner.java:94)
at org.junit.runners.ParentRunner.run(ParentRunner.java:290)
at org.junit.runners.ParentRunner.schedule(ParentRunner.java:71)
at org.junit.runners.ParentRunner.runChildren(ParentRunner.java:288)
at org.junit.runners.ParentRunner.access[=12=]0(ParentRunner.java:58)
at org.junit.runners.ParentRunner.evaluate(ParentRunner.java:268)
at org.springframework.test.context.junit4.statements.RunBeforeTestClassCallbacks.
现在 Bibucket 管道使用 Docker 图像 (java:8)。当我在本地切换回 @SpringBoot
和 @AutoConfigureMockMvc
时,我在两种环境中都遇到了相同的错误。两个环境中的数据库设置相同(MongoDB 的 docker 图像相同),所有内容都相同...可能是某些端口在使用 Docker 时未被映射?我想我确实创建了 Servlet 请求...
编辑
在 Docker 容器中模拟 Bitbucket 管道构建(如建议 here),似乎模拟 MappingMongoConverter
并与 @SpringBootTest
一起移动 @AutoConfigureMockMvc
就够了运行ning。因此 @WebMvcTest
只有部分模拟出的上下文在没有容器的情况下就足够了,但它会在 Docker 容器内失败,例如使用 Bitbucket 时出现的容器。为什么?
原来缺少一些关键 bean,因为 @WebMvc
注释不会获取 @Components
,只有 Web 堆栈需要的东西不包括存储库,但问题是我的安全配置我也想对其进行测试,现在我只是 @Import
(以及控制器所依赖的其他一些 bean):
@RunWith(SpringRunner.class)
@WebMvcTest(value = VZNFCController.class)
@Import(value = {VZApiSecurityConfiguration.class,
VZJwtTokenUtils.class, VZProperties.class})
public class VZNFCControllerTest {
@Autowired
private MockMvc mvc;
@MockBean
private VZNFCTagService tagService;
/* same as the above... I don't mock out the repository any more */
}
这里学到的教训是,所有自动配置的测试上下文(@WebMvc
、@DataMongoTest
等)都会加快您的测试速度(这很好,因为我在 bitbucket 上支付了构建时间)。但是您需要真正了解获取应用程序的那一部分需要什么 运行。它迫使我真的只模拟服务以专注于控制器,然后为我的应用程序的 DAO 部分编写更多测试。我想这是件好事。
当 运行在本地 @WebMvcTest
时,我没有问题(Spring Boot 1.5.8,gradle 4.6):
@RunWith(SpringRunner.class)
@WebMvcTest(VZNFCController.class)
public class VZNFCControllerTest {
private VZNFCTagAction action1 = new VZNFCTagAction();
private VZNFCTagAction action2 = new VZNFCTagAction();
@Before
public void setUp(){
action1.setMessage("message1");
action2.setMessage("message2");
}
@Autowired
private MockMvc mvc;
@MockBean
private VZNFCTagActionRepository actionRepository;
@MockBean
private MappingMongoConverter mongoConverter;
@Test
@WithMockUser
public void testGetTagList() throws Exception {
given(this.actionRepository.findAll())
.willReturn(Arrays.asList(action1, action2));
this.mvc.perform(get("/api/nfc/tags")
.accept(MediaType.APPLICATION_JSON_UTF8))
.andExpect(status().isOk());
}
}
但是,当我上传到 Atlassian Bitbucket 并在那里 运行 ./gradlew test --stacktrace
时,我得到以下信息:
java.lang.IllegalStateException: Failed to load ApplicationContext
at org.springframework.test.context.cache.DefaultCacheAwareContextLoaderDelegate.loadContext(DefaultCacheAwareContextLoaderDelegate.java:124)
at org.springframework.test.context.support.DefaultTestContext.getApplicationContext(DefaultTestContext.java:83)
at org.springframework.test.context.support.DependencyInjectionTestExecutionListener.injectDependencies(DependencyInjectionTestExecutionListener.java:117)
at org.springframework.test.context.support.DependencyInjectionTestExecutionListener.prepareTestInstance(DependencyInjectionTestExecutionListener.java:83)
at org.springframework.boot.test.autoconfigure.SpringBootDependencyInjectionTestExecutionListener.prepareTestInstance(SpringBootDependencyInjectionTestExecutionListener.java:44)
at org.springframework.test.context.TestContextManager.prepareTestInstance(TestContextManager.java:230)
at org.springframework.test.context.junit4.SpringJUnit4ClassRunner.createTest(SpringJUnit4ClassRunner.java:228)
at org.springframework.test.context.junit4.SpringJUnit4ClassRunner.runReflectiveCall(SpringJUnit4ClassRunner.java:287)
at org.junit.internal.runners.model.ReflectiveCallable.run(ReflectiveCallable.java:12)
at org.springframework.test.context.junit4.SpringJUnit4ClassRunner.methodBlock(SpringJUnit4ClassRunner.java:289)
at org.springframework.test.context.junit4.SpringJUnit4ClassRunner.runChild(SpringJUnit4ClassRunner.java:247)
at org.springframework.test.context.junit4.SpringJUnit4ClassRunner.runChild(SpringJUnit4ClassRunner.java:94)
at org.junit.runners.ParentRunner.run(ParentRunner.java:290)
at org.junit.runners.ParentRunner.schedule(ParentRunner.java:71)
at org.junit.runners.ParentRunner.runChildren(ParentRunner.java:288)
at org.junit.runners.ParentRunner.access[=12=]0(ParentRunner.java:58)
at org.junit.runners.ParentRunner.evaluate(ParentRunner.java:268)
at org.springframework.test.context.junit4.statements.RunBeforeTestClassCallbacks.
现在 Bibucket 管道使用 Docker 图像 (java:8)。当我在本地切换回 @SpringBoot
和 @AutoConfigureMockMvc
时,我在两种环境中都遇到了相同的错误。两个环境中的数据库设置相同(MongoDB 的 docker 图像相同),所有内容都相同...可能是某些端口在使用 Docker 时未被映射?我想我确实创建了 Servlet 请求...
编辑
在 Docker 容器中模拟 Bitbucket 管道构建(如建议 here),似乎模拟 MappingMongoConverter
并与 @SpringBootTest
一起移动 @AutoConfigureMockMvc
就够了运行ning。因此 @WebMvcTest
只有部分模拟出的上下文在没有容器的情况下就足够了,但它会在 Docker 容器内失败,例如使用 Bitbucket 时出现的容器。为什么?
原来缺少一些关键 bean,因为 @WebMvc
注释不会获取 @Components
,只有 Web 堆栈需要的东西不包括存储库,但问题是我的安全配置我也想对其进行测试,现在我只是 @Import
(以及控制器所依赖的其他一些 bean):
@RunWith(SpringRunner.class)
@WebMvcTest(value = VZNFCController.class)
@Import(value = {VZApiSecurityConfiguration.class,
VZJwtTokenUtils.class, VZProperties.class})
public class VZNFCControllerTest {
@Autowired
private MockMvc mvc;
@MockBean
private VZNFCTagService tagService;
/* same as the above... I don't mock out the repository any more */
}
这里学到的教训是,所有自动配置的测试上下文(@WebMvc
、@DataMongoTest
等)都会加快您的测试速度(这很好,因为我在 bitbucket 上支付了构建时间)。但是您需要真正了解获取应用程序的那一部分需要什么 运行。它迫使我真的只模拟服务以专注于控制器,然后为我的应用程序的 DAO 部分编写更多测试。我想这是件好事。