在应用程序启动之前配置@MockBean组件

我有一个Spring Boot 1.4.2应用程序。启动期间使用的一些代码如下所示:

@Component 
class SystemTypeDetector{
    public enum SystemType{ TYPE_A, TYPE_B, TYPE_C }
    public SystemType getSystemType(){ return ... }
}

@Component 
public class SomeOtherComponent{
    @Autowired 
    private SystemTypeDetector systemTypeDetector;
    @PostConstruct 
    public void startup(){
        switch(systemTypeDetector.getSystemType()){   // <-- NPE here in test
        case TYPE_A: ...
        case TYPE_B: ...
        case TYPE_C: ...
        }
    }
}

有一个组件用于确定系统类型。此组件在从其他组件启动期间使用。在生产中,一切正常。

现在我想使用Spring 1.4添加一些集成测试。@MockBean

测试如下所示:

@RunWith(SpringRunner.class)
@SpringBootTest(classes = MyWebApplication.class, webEnvironment = RANDOM_PORT)
public class IntegrationTestNrOne {
    @MockBean 
    private SystemTypeDetector systemTypeDetectorMock;

    @Before 
    public void initMock(){
       Mockito.when(systemTypeDetectorMock.getSystemType()).thenReturn(TYPE_C);
    }

    @Test 
    public void testNrOne(){
      // ...
    }
}

基本上,嘲笑工作正常。我的 systemTypeDetectorMock 被使用,如果我调用 -> 返回。getSystemTypeTYPE_C

问题是应用程序无法启动。目前弹簧工作顺序似乎是:

  1. 创建所有模拟(不带配置,所有方法都返回 null)
  2. 开始申请
  3. 调用@Before方法(将配置模拟的位置)
  4. 开始测试

我的问题是应用程序以未初始化的模拟开始。因此,对 的调用返回 null。getSystemType()

我的问题是:如何在应用程序启动之前配置模拟?

编辑:如果有人遇到同样的问题,一种解决方法是使用 。这调用了真正的组件,在我的情况下,系统启动。启动后,我可以更改模拟行为。@MockBean(answer = CALLS_REAL_METHODS)


答案 1

在这种情况下,您需要以我们在引入之前使用的方式配置模拟 - 通过手动指定一个将替换上下文中原始bean的bean。@MockBean@Primary

@SpringBootTest
class DemoApplicationTests {

    @TestConfiguration
    public static class TestConfig {

        @Bean
        @Primary
        public SystemTypeDetector mockSystemTypeDetector() {
            SystemTypeDetector std = mock(SystemTypeDetector.class);
            when(std.getSystemType()).thenReturn(TYPE_C);
            return std;
        }

    }

    @Autowired
    private SystemTypeDetector systemTypeDetector;

    @Test
    void contextLoads() {
        assertThat(systemTypeDetector.getSystemType()).isEqualTo(TYPE_C);
    }
}

由于类是静态内部类,因此只有此测试才会自动选择它。您将放入的完整模拟行为必须移动到初始化Bean的方法。@TestConfiguration@Before


答案 2

我能够像这样修复它

@RunWith(SpringRunner.class)
@SpringBootTest(classes = MyWebApplication.class, webEnvironment = RANDOM_PORT)
public class IntegrationTestNrOne {
    // this inner class must be static!
    @TestConfiguration
    public static class EarlyConfiguration {
       @MockBean 
       private SystemTypeDetector systemTypeDetectorMock;

       @PostConstruct 
       public void initMock(){
          Mockito.when(systemTypeDetectorMock.getSystemType()).thenReturn(TYPE_C);
       }
    }

    // here we can inject the bean created by EarlyConfiguration
    @Autowired 
    private SystemTypeDetector systemTypeDetectorMock;

    @Autowired
    private SomeOtherComponent someOtherComponent;

    @Test 
    public void testNrOne(){
       someOtherComponent.doStuff();
    }
}

推荐