这种测试方案会加载完整的SpringContext,但我们仍然不需要Web Server,需要继续通过MockMVC来模拟请求。
在测试的时候主要用到了 @ SpringBootTest 注解。看下代码:
@RunWith(SpringRunner.class) @SpringBootTest @AutoConfigureMockMvc public class WorkerControllerSpringBootMockTest { @Autowired private MockMvc mockMvc; private JacksonTester<Worker> jsonWorker; @Before public void setup() { JacksonTester.initFields(this, new ObjectMapper()); System.out.println(); } @Test public void getWhenExists() throws Exception { //when MockHttpServletResponse response = mockMvc.perform(get("/worker/2").accept(MediaType.APPLICATION_JSON)) .andReturn().getResponse(); //then assertThat(response.getStatus()).isEqualTo(HttpStatus.OK.value()); assertThat(response.getContentAsString()).isEqualTo(jsonWorker.write(new Worker("raccoon", 23)).getJson()); } @Test //@DirtiesContext public void getWhenNotExists() throws Exception { //when MockHttpServletResponse response = mockMvc.perform(get("/worker/26").accept(MediaType.APPLICATION_JSON)) .andReturn().getResponse(); //then assertThat(response.getStatus()).isEqualTo(HttpStatus.NOT_FOUND.value()); assertThat(response.getContentAsString()).isEmpty(); } @Test public void getByNameWhenExists() throws Exception { //when MockHttpServletResponse response = mockMvc.perform(get("/worker?name=HanMeimei").accept(MediaType.APPLICATION_JSON)) .andReturn().getResponse(); //then assertThat(response.getStatus()).isEqualTo(HttpStatus.OK.value()); assertThat(response.getContentAsString()).isEqualTo(jsonWorker.write(new Worker("HanMeimei", 16)).getJson()); } @Test public void getByNameWhenNotExists() throws Exception { //when MockHttpServletResponse response = mockMvc.perform(get("/worker?name=LiLei").accept(MediaType.APPLICATION_JSON)) .andReturn().getResponse(); //then assertThat(response.getStatus()).isEqualTo(HttpStatus.OK.value()); assertThat(response.getContentAsString()).isEmpty(); } @Test public void add() throws Exception { MockHttpServletResponse response = mockMvc.perform( post("/worker").contentType(MediaType.APPLICATION_JSON) .content(jsonWorker.write(new Worker("Jerry", 12)).getJson()) ).andReturn().getResponse(); assertThat(response.getStatus()).isEqualTo(HttpStatus.CREATED.value()); } @Test public void workerFilter() throws Exception { //when MockHttpServletResponse response = mockMvc.perform(get("/worker/2").accept(MediaType.APPLICATION_JSON)) .andReturn().getResponse(); //then assertThat(response.getStatus()).isEqualTo(HttpStatus.OK.value()); assertThat(response.getHeaders("X-CHOBIT-APP")).containsOnly("chobit-header"); } }
使用 @ SpringBootTest 注解会加载整个Context。这样我们可以自动获得所有在Context中注入的Bean,以及从application.properties中加载的配置信息。
在 @ SpringBootTest 中声明webEnvironment为 WebEnvironment . MOCK (默认值就是 WebEnvironment . MOCK )后,结合 @ AutoConfigureMockMvc 注解,在测试的时候会得到一个模拟的Web/Servlet环境。
因为没有Web Server,所以就无法使用 RestTemplate ,也就只能继续使用 MockMVC 了。这次 MockMVC 的实例是由 @ AutoConfigureMockMvc 注解来完成的。这归功于SpringBoot的自动化配置。
所有参与测试的对象的关系如下图:
这种测试方案更倾向于集成测试。它的关注点主要在于SpringBoot不同类之间的交互。
在这个测试方案中,请求仍然是通过MockMVC模拟的。不过因为有一个完整的Context,请求处理过程中的所有逻辑都是真实的,请求返回结果也是真实的。因此测试效果和使用Web Server几乎是差不多的了。
如果还是要只测试Controller中的逻辑,也可以继续使用 @ MockBean 注解来mock一个 IWorkerService 实例来覆盖Context中已有的实例。即使使用了真正的WebServer,也可以继续使用MockBean。同样在测试中也需要继续mock数据。
概括来说,这种测试的位置稍显尴尬:如果要执行单元测试来测试WEB层的逻辑,建议优先第二种方案;如果要执行集成测试,启动一个真正的Web Server,使用 RestTemplate 进行测试会更彻底也更加方便。
其他:示例代码可在CSDN下载,地址:https://download.csdn.net/download/tianxiexingyun/11065824