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 部分编写更多测试。我想这是件好事。