隨着RESTful Web Service的流行,測試對外的Service是否知足指望也變的必要的。從Spring 3.2開始Spring了Spring Web測試框架,若是版本低於3.2,請使用spring-test-mvc項目(合併到spring3.2中了)。javascript
Spring MVC測試框架提供了對服務器端和客戶端(基於RestTemplate的客戶端)提供了支持。java
對於服務器端:在Spring 3.2以前,咱們測試時通常都是直接new控制器,注入依賴,而後判斷返回值。可是咱們沒法連同Spring MVC的基礎設施(如DispatcherServlet調度、類型轉換、數據綁定、攔截器等)一塊兒測試,另外也沒有現成的方法測試如最終渲染的視圖(@ResponseBody生成的JSON/XML、JSP、Velocity等)內容是否正確。從Spring 3.2開始這些事情均可以完成了。並且能夠測試完整的Spring MVC流程,即從URL請求到控制器處理,再到視圖渲染均可以測試。git
對於客戶端:不須要啓動服務器便可測試咱們的RESTful 服務。github
1 服務器端測試
個人環境:JDK七、Maven三、spring四、Servlet3web
首先添加依賴
以下是spring-context和spring-webmvc依賴:spring
Java代碼 express
- <dependency>
- <groupId>org.springframework</groupId>
- <artifactId>spring-context</artifactId>
- <version>${spring.version}</version>
- </dependency>
-
- <dependency>
- <groupId>org.springframework</groupId>
- <artifactId>spring-webmvc</artifactId>
- <version>${spring.version}</version>
- </dependency>
版本信息:<spring.version>4.0.0.RELEASE</spring.version>json
以下是測試相關的依賴(junit、hamcrest、mockito、spring-test):spring-mvc
Java代碼 安全
- <dependency>
- <groupId>junit</groupId>
- <artifactId>junit</artifactId>
- <version>${junit.version}</version>
- <scope>test</scope>
- </dependency>
-
- <dependency>
- <groupId>org.hamcrest</groupId>
- <artifactId>hamcrest-core</artifactId>
- <version>${hamcrest.core.version}/version>
- <scope>test</scope>
- </dependency>
- <dependency>
- <groupId>org.mockito</groupId>
- <artifactId>mockito-core</artifactId>
- <version>${mockito.core.version}</version>
- <scope>test</scope>
- </dependency>
-
- <dependency>
- <groupId>org.springframework</groupId>
- <artifactId>spring-test</artifactId>
- <version>${spring.version}</version>
- <scope>test</scope>
- </dependency>
版本信息:<junit.version>4.11</junit.version>、<hamcrest.core.version>1.3</hamcrest.core.version>、<mockito.core.version>1.9.5</mockito.core.version>
而後準備測試相關配置
實體:
Java代碼
- package com.sishuok.mvc.entity;
- import java.io.Serializable;
- public class User implements Serializable {
- private Long id;
- private String name;
- //省略getter/setter等
- }
控制器:
Java代碼
- package com.sishuok.mvc.controller;
- //省略import
- @Controller
- @RequestMapping("/user")
- public class UserController {
-
- @RequestMapping("/{id}")
- public ModelAndView view(@PathVariable("id") Long id, HttpServletRequest req) {
- User user = new User();
- user.setId(id);
- user.setName("zhang");
-
- ModelAndView mv = new ModelAndView();
- mv.addObject("user", user);
- mv.setViewName("user/view");
- return mv;
- }
- }
XML風格配置:
spring-config.xml:加載非web層組件
Java代碼
- <?xml version="1.0" encoding="UTF-8"?>
- <beans xmlns="http://www.springframework.org/schema/beans"
- xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
- xmlns:context="http://www.springframework.org/schema/context"
- xsi:schemaLocation="
- http://www.springframework.org/schema/beans http://www.springframework.org/schema/beans/spring-beans.xsd
- http://www.springframework.org/schema/context http://www.springframework.org/schema/context/spring-context.xsd
- ">
- <!-- 經過web.xml中的 org.springframework.web.context.ContextLoaderListener 加載的 -->
- <!-- 請參考 http://jinnianshilongnian.iteye.com/blog/1602617 -->
- <context:component-scan base-package="com.sishuok.mvc">
- <context:exclude-filter type="annotation" expression="org.springframework.stereotype.Controller"/>
- </context:component-scan>
- </beans>
spring-mvc.xml:加載和配置web層組件
Java代碼
- <?xml version="1.0" encoding="UTF-8"?>
- <beans xmlns="http://www.springframework.org/schema/beans"
- xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
- xmlns:context="http://www.springframework.org/schema/context"
- xmlns:mvc="http://www.springframework.org/schema/mvc"
- xsi:schemaLocation="
- http://www.springframework.org/schema/beans http://www.springframework.org/schema/beans/spring-beans.xsd
- http://www.springframework.org/schema/context http://www.springframework.org/schema/context/spring-context.xsd
- http://www.springframework.org/schema/mvc http://www.springframework.org/schema/mvc/spring-mvc.xsd
- ">
- <!-- 經過web.xml中的 org.springframework.web.servlet.DispatcherServlet 加載的 -->
- <!-- 請參考 http://jinnianshilongnian.iteye.com/blog/1602617 -->
- <context:component-scan base-package="com.sishuok.mvc" use-default-filters="false">
- <context:include-filter type="annotation" expression="org.springframework.stereotype.Controller"/>
- </context:component-scan>
- <mvc:annotation-driven/>
- <bean id="viewResolver" class="org.springframework.web.servlet.view.InternalResourceViewResolver">
- <property name="prefix" value="/WEB-INF/jsp/"/>
- <property name="suffix" value=".jsp"/>
- </bean>
- </beans>
web.xml配置:此處就不貼了,請前往github查看。
對於context:component-scan注意事項請參考《context:component-scan掃描使用上的容易忽略的use-default-filters》和《第三章 DispatcherServlet詳解 ——跟開濤學SpringMVC》。
等價的註解風格配置:
AppConfig.java:等價於spring-config.xml
Java代碼
- package com.sishuok.config;
-
- import org.springframework.context.annotation.ComponentScan;
- import org.springframework.context.annotation.Configuration;
- import org.springframework.context.annotation.FilterType;
- import org.springframework.stereotype.Controller;
-
- @Configuration
- @ComponentScan(basePackages = "com.sishuok.mvc", excludeFilters = {
- @ComponentScan.Filter(type = FilterType.ANNOTATION, value = {Controller.class})
- })
- public class AppConfig {
- }
MvcConfig.java:等價於spring-mvc.xml
Java代碼
- package com.sishuok.config;
-
- import org.springframework.context.annotation.Bean;
- import org.springframework.context.annotation.ComponentScan;
- import org.springframework.context.annotation.Configuration;
- import org.springframework.context.annotation.FilterType;
- import org.springframework.stereotype.Controller;
- import org.springframework.web.servlet.ViewResolver;
- import org.springframework.web.servlet.config.annotation.EnableWebMvc;
- import org.springframework.web.servlet.config.annotation.WebMvcConfigurationSupport;
- import org.springframework.web.servlet.view.InternalResourceViewResolver;
-
- @Configuration
- @EnableWebMvc
- @ComponentScan(basePackages = "com.sishuok.mvc", useDefaultFilters = false, includeFilters = {
- @ComponentScan.Filter(type = FilterType.ANNOTATION, value = {Controller.class})
- })
- public class MvcConfig extends WebMvcConfigurationSupport {
-
- @Bean
- public ViewResolver viewResolver() {
- InternalResourceViewResolver viewResolver = new InternalResourceViewResolver();
- viewResolver.setPrefix("/WEB-INF/jsp/");
- viewResolver.setSuffix(".jsp");
- return viewResolver;
- }
-
- }
WebInitializer.java:註冊相應的web.xml中的組件
Java代碼
- package com.sishuok.config;
-
- import org.springframework.web.WebApplicationInitializer;
- import org.springframework.web.context.ContextLoaderListener;
- import org.springframework.web.context.support.AnnotationConfigWebApplicationContext;
- import org.springframework.web.filter.CharacterEncodingFilter;
- import org.springframework.web.servlet.DispatcherServlet;
-
- import javax.servlet.DispatcherType;
- import javax.servlet.FilterRegistration;
- import javax.servlet.ServletException;
- import javax.servlet.ServletRegistration;
- import java.util.EnumSet;
-
- public class WebInitializer implements WebApplicationInitializer {
-
- @Override
- public void onStartup(javax.servlet.ServletContext sc) throws ServletException {
-
- AnnotationConfigWebApplicationContext rootContext = new AnnotationConfigWebApplicationContext();
- rootContext.register(AppConfig.class);
- sc.addListener(new ContextLoaderListener(rootContext));
-
- //二、springmvc上下文
- AnnotationConfigWebApplicationContext springMvcContext = new AnnotationConfigWebApplicationContext();
- springMvcContext.register(MvcConfig.class);
- //三、DispatcherServlet
- DispatcherServlet dispatcherServlet = new DispatcherServlet(springMvcContext);
- ServletRegistration.Dynamic dynamic = sc.addServlet("dispatcherServlet", dispatcherServlet);
- dynamic.setLoadOnStartup(1);
- dynamic.addMapping("/");
-
- //四、CharacterEncodingFilter
- CharacterEncodingFilter characterEncodingFilter = new CharacterEncodingFilter();
- characterEncodingFilter.setEncoding("utf-8");
- FilterRegistration filterRegistration =
- sc.addFilter("characterEncodingFilter", characterEncodingFilter);
- filterRegistration.addMappingForUrlPatterns(EnumSet.of(DispatcherType.REQUEST), false, "/");
-
- }
- }
對於WebInitializer,請參考《Spring4新特性——Groovy Bean定義DSL》
到此基本的配置就搞定了,接下來看看如何測試吧。
1.1 之前的測試方式
Java代碼
- package com.sishuok.mvc.controller;
- //省略import
- public class UserControllerTest {
-
- private UserController userController;
-
- @Before
- public void setUp() {
- userController = new UserController();
- //安裝userCtroller依賴 好比userService
- }
-
- @Test
- public void testView() {
- MockHttpServletRequest req = new MockHttpServletRequest();
- ModelAndView mv = userController.view(1L, req);
-
- ModelAndViewAssert.assertViewName(mv, "user/view");
- ModelAndViewAssert.assertModelAttributeAvailable(mv, "user");
-
- }
- }
準備控制器:咱們經過new方式建立一個,而後手工查找依賴注入進去(好比從spring容器獲取/new的);
Mock Request:此處使用Spring提供的Mock API模擬一個HttpServletRequest,其餘的Servlet API也提供了相應的Mock類,具體請查看Javadoc;
訪問控制器方法:經過直接調用控制器方法進行訪問,此處沒法驗證Spring MVC框架的類型轉換、數據驗證等是否正常;
ModelAndViewAssert:經過這個Assert API驗證咱們的返回值是否正常;
對於單元測試步驟請參考:加速Java應用開發速度3——單元/集成測試+CI
這種方式的缺點已經說過了,如不能走Spring MVC完整流程(不能走Servlet的過濾器鏈、SpringMVC的類型轉換、數據驗證、數據綁定、攔截器等等),若是作基本的測試沒問題,這種方式就是純粹的單元測試,咱們想要的功能實際上是一種集成測試,不事後續部分不區分。
1.2 安裝測試環境
spring mvc測試框架提供了兩種方式,獨立安裝和集成Web環境測試(此種方式並不會集成真正的web環境,而是經過相應的Mock API進行模擬測試,無須啓動服務器)。
獨立測試方式
Java代碼
- public class UserControllerStandaloneSetupTest {
- private MockMvc mockMvc;
- @Before
- public void setUp() {
- UserController userController = new UserController();
- mockMvc = MockMvcBuilders.standaloneSetup(userController).build();
- }
- }
一、首先本身建立相應的控制器,注入相應的依賴
二、經過MockMvcBuilders.standaloneSetup模擬一個Mvc測試環境,經過build獲得一個MockMvc
三、MockMvc:是咱們之後測試時常用的API,後邊介紹
集成Web環境方式
Java代碼
- //XML風格
- @RunWith(SpringJUnit4ClassRunner.class)
- @WebAppConfiguration(value = "src/main/webapp")
- @ContextHierarchy({
- @ContextConfiguration(name = "parent", locations = "classpath:spring-config.xml"),
- @ContextConfiguration(name = "child", locations = "classpath:spring-mvc.xml")
- })
-
- //註解風格
- //@RunWith(SpringJUnit4ClassRunner.class)
- //@WebAppConfiguration(value = "src/main/webapp")
- //@ContextHierarchy({
- // @ContextConfiguration(name = "parent", classes = AppConfig.class),
- // @ContextConfiguration(name = "child", classes = MvcConfig.class)
- //})
- public class UserControllerWebAppContextSetupTest {
-
- @Autowired
- private WebApplicationContext wac;
- private MockMvc mockMvc;
-
- @Before
- public void setUp() {
- mockMvc = MockMvcBuilders.webAppContextSetup(wac).build();
- }
- }
一、@WebAppConfiguration:測試環境使用,用來表示測試環境使用的ApplicationContext將是WebApplicationContext類型的;value指定web應用的根;
二、@ContextHierarchy:指定容器層次,即spring-config.xml是父容器,而spring-mvc.xml是子容器,請參考《第三章 DispatcherServlet詳解 ——跟開濤學SpringMVC》
三、經過@Autowired WebApplicationContext wac:注入web環境的ApplicationContext容器;
四、而後經過MockMvcBuilders.webAppContextSetup(wac).build()建立一個MockMvc進行測試;
到此測試環境就搭建完成了,根據須要選擇使用哪一種方式便可。相關配置請前往github查看。
1.三、HelloWorld
Java代碼
- @Test
- public void testView() throws Exception {
- MvcResult result = mockMvc.perform(MockMvcRequestBuilders.get("/user/1"))
- .andExpect(MockMvcResultMatchers.view().name("user/view"))
- .andExpect(MockMvcResultMatchers.model().attributeExists("user"))
- .andDo(MockMvcResultHandlers.print())
- .andReturn();
-
- Assert.assertNotNull(result.getModelAndView().getModel().get("user"));
- }
一、mockMvc.perform執行一個請求;
二、MockMvcRequestBuilders.get("/user/1")構造一個請求
三、ResultActions.andExpect添加執行完成後的斷言
四、ResultActions.andDo添加一個結果處理器,表示要對結果作點什麼事情,好比此處使用MockMvcResultHandlers.print()輸出整個響應結果信息。
五、ResultActions.andReturn表示執行完成後返回相應的結果。
整個測試過程很是有規律:
一、準備測試環境
二、經過MockMvc執行請求
3.一、添加驗證斷言
3.二、添加結果處理器
3.三、獲得MvcResult進行自定義斷言/進行下一步的異步請求
四、卸載測試環境
對於單元測試步驟請參考:加速Java應用開發速度3——單元/集成測試+CI。
1.四、瞭解測試API
Spring mvc測試框架提供了測試MVC須要的API,主要包括Servlet/JSP Mock、MockMvcBuilder、MockMvc、RequestBuilder、ResultMatcher、ResultHandler、MvcResult等。另外提供了幾個靜態工廠方法便於測試:MockMvcBuilders、MockMvcRequestBuilders、MockMvcResultMatchers、MockMvcResultHandlers。在使用時請使用靜態方法導入方便測試,如:
Java代碼
- import static org.springframework.test.web.servlet.setup.MockMvcBuilders.*;
- import static org.springframework.test.web.servlet.request.MockMvcRequestBuilders.*;
- import static org.springframework.test.web.servlet.result.MockMvcResultMatchers.*;
- import static org.springframework.test.web.servlet.result.MockMvcResultHandlers.*;
Servlet/JSP API Mock
提供了對Servlet 3 相應API的Mock,如:
MockServletContext
MockHttpServletRequest
MockHttpServletResponse
……
具體請查看spring-test模塊的org.springframework.mock.web包。
MockMvcBuilder/MockMvcBuilders
MockMvcBuilder是用來構造MockMvc的構造器,其主要有兩個實現:StandaloneMockMvcBuilder和DefaultMockMvcBuilder,分別對應以前的兩種測試方式。對於咱們來講直接使用靜態工廠MockMvcBuilders建立便可:
MockMvcBuilders.webAppContextSetup(WebApplicationContext context):指定WebApplicationContext,將會從該上下文獲取相應的控制器並獲得相應的MockMvc;
MockMvcBuilders.standaloneSetup(Object... controllers):經過參數指定一組控制器,這樣就不須要從上下文獲取了;
其中DefaultMockMvcBuilder還提供了以下API:
addFilters(Filter... filters)/addFilter(Filter filter, String... urlPatterns):添加javax.servlet.Filter過濾器
defaultRequest(RequestBuilder requestBuilder):默認的RequestBuilder,每次執行時會合併到自定義的RequestBuilder中,即提供公共請求數據的;
alwaysExpect(ResultMatcher resultMatcher):定義全局的結果驗證器,即每次執行請求時都進行驗證的規則;
alwaysDo(ResultHandler resultHandler):定義全局結果處理器,即每次請求時都進行結果處理;
dispatchOptions:DispatcherServlet是否分發OPTIONS請求方法到控制器;
StandaloneMockMvcBuilder繼承了DefaultMockMvcBuilder,又提供了以下API:
setMessageConverters(HttpMessageConverter<?>...messageConverters):設置HTTP消息轉換器;
setValidator(Validator validator):設置驗證器;
setConversionService(FormattingConversionService conversionService):設置轉換服務;
addInterceptors(HandlerInterceptor... interceptors)/addMappedInterceptors(String[] pathPatterns, HandlerInterceptor... interceptors):添加spring mvc攔截器;
setContentNegotiationManager(ContentNegotiationManager contentNegotiationManager):設置內容協商管理器;
setAsyncRequestTimeout(long timeout):設置異步超時時間;
setCustomArgumentResolvers(HandlerMethodArgumentResolver... argumentResolvers):設置自定義控制器方法參數解析器;
setCustomReturnValueHandlers(HandlerMethodReturnValueHandler... handlers):設置自定義控制器方法返回值處理器;
setHandlerExceptionResolvers(List<HandlerExceptionResolver> exceptionResolvers)/setHandlerExceptionResolvers(HandlerExceptionResolver... exceptionResolvers):設置異常解析器;
setViewResolvers(ViewResolver...resolvers):設置視圖解析器;
setSingleView(View view):設置單個視圖,即視圖解析時老是解析到這一個(僅適用於只有一個視圖的狀況);
setLocaleResolver(LocaleResolver localeResolver):設置Local解析器;
setFlashMapManager(FlashMapManager flashMapManager):設置FlashMapManager,如存儲重定向數據;
setUseSuffixPatternMatch(boolean useSuffixPatternMatch):設置是不是後綴模式匹配,如「/user」是否匹配"/user.*",默認真即匹配;
setUseTrailingSlashPatternMatch(boolean useTrailingSlashPatternMatch):設置是否自動後綴路徑模式匹配,如「/user」是否匹配「/user/」,默認真即匹配;
addPlaceHolderValue(String name, String value) :添加request mapping中的佔位符替代;
由於StandaloneMockMvcBuilder不會加載Spring MVC配置文件,所以就不會註冊咱們須要的一些組件,所以就提供瞭如上API用於註冊咱們須要的相應組件。
MockMvc
使用以前的MockMvcBuilder.build()獲得構建好的MockMvc;這個是mvc測試的核心API,對於該API的使用方式以下:
Java代碼
- MvcResult result = mockMvc.perform(MockMvcRequestBuilders.get("/user/1"))
- .andExpect(MockMvcResultMatchers.view().name("user/view"))
- .andExpect(MockMvcResultMatchers.model().attributeExists("user"))
- .andDo(MockMvcResultHandlers.print())
- .andReturn();
perform:執行一個RequestBuilder請求,會自動執行SpringMVC的流程並映射到相應的控制器執行處理;
andExpect:添加ResultMatcher驗證規則,驗證控制器執行完成後結果是否正確;
andDo:添加ResultHandler結果處理器,好比調試時打印結果到控制檯;
andReturn:最後返回相應的MvcResult;而後進行自定義驗證/進行下一步的異步處理;
另外還提供瞭如下API:
setDefaultRequest:設置默認的RequestBuilder,用於在每次perform執行相應的RequestBuilder時自動把該默認的RequestBuilder合併到perform的RequestBuilder中;
setGlobalResultMatchers:設置全局的預期結果驗證規則,如咱們經過MockMvc測試多個控制器時,假設它們都想驗證某個規則時,就可使用這個;
setGlobalResultHandlers:設置全局的ResultHandler結果處理器;
RequestBuilder/MockMvcRequestBuilders
從名字能夠看出,RequestBuilder用來構建請求的,其提供了一個方法buildRequest(ServletContext servletContext)用於構建MockHttpServletRequest;其主要有兩個子類MockHttpServletRequestBuilder和MockMultipartHttpServletRequestBuilder(如文件上傳使用),即用來Mock客戶端請求須要的全部數據。
MockMvcRequestBuilders主要API:
MockHttpServletRequestBuilder get(String urlTemplate, Object... urlVariables):根據uri模板和uri變量值獲得一個GET請求方式的MockHttpServletRequestBuilder;如get("/user/{id}", 1L);
MockHttpServletRequestBuilder post(String urlTemplate, Object... urlVariables):同get相似,可是是POST方法;
MockHttpServletRequestBuilder put(String urlTemplate, Object... urlVariables):同get相似,可是是PUT方法;
MockHttpServletRequestBuilder delete(String urlTemplate, Object... urlVariables) :同get相似,可是是DELETE方法;
MockHttpServletRequestBuilder options(String urlTemplate, Object... urlVariables):同get相似,可是是OPTIONS方法;
MockHttpServletRequestBuilder request(HttpMethod httpMethod, String urlTemplate, Object... urlVariables):提供本身的Http請求方法及uri模板和uri變量,如上API都是委託給這個API;
MockMultipartHttpServletRequestBuilder fileUpload(String urlTemplate, Object... urlVariables):提供文件上傳方式的請求,獲得MockMultipartHttpServletRequestBuilder;
RequestBuilder asyncDispatch(final MvcResult mvcResult):建立一個從啓動異步處理的請求的MvcResult進行異步分派的RequestBuilder;
接下來再看看MockHttpServletRequestBuilder和MockMultipartHttpServletRequestBuilder API:
MockHttpServletRequestBuilder API:
MockHttpServletRequestBuilder header(String name, Object... values)/MockHttpServletRequestBuilder headers(HttpHeaders httpHeaders):添加頭信息;
MockHttpServletRequestBuilder contentType(MediaType mediaType):指定請求的contentType頭信息;
MockHttpServletRequestBuilder accept(MediaType... mediaTypes)/MockHttpServletRequestBuilder accept(String... mediaTypes):指定請求的Accept頭信息;
MockHttpServletRequestBuilder content(byte[] content)/MockHttpServletRequestBuilder content(String content):指定請求Body體內容;
MockHttpServletRequestBuilder cookie(Cookie... cookies):指定請求的Cookie;
MockHttpServletRequestBuilder locale(Locale locale):指定請求的Locale;
MockHttpServletRequestBuilder characterEncoding(String encoding):指定請求字符編碼;
MockHttpServletRequestBuilder requestAttr(String name, Object value) :設置請求屬性數據;
MockHttpServletRequestBuilder sessionAttr(String name, Object value)/MockHttpServletRequestBuilder sessionAttrs(Map<String, Object> sessionAttributes):設置請求session屬性數據;
MockHttpServletRequestBuilder flashAttr(String name, Object value)/MockHttpServletRequestBuilder flashAttrs(Map<String, Object> flashAttributes):指定請求的flash信息,好比重定向後的屬性信息;
MockHttpServletRequestBuilder session(MockHttpSession session) :指定請求的Session;
MockHttpServletRequestBuilder principal(Principal principal) :指定請求的Principal;
MockHttpServletRequestBuilder contextPath(String contextPath) :指定請求的上下文路徑,必須以「/」開頭,且不能以「/」結尾;
MockHttpServletRequestBuilder pathInfo(String pathInfo) :請求的路徑信息,必須以「/」開頭;
MockHttpServletRequestBuilder secure(boolean secure):請求是否使用安全通道;
MockHttpServletRequestBuilder with(RequestPostProcessor postProcessor):請求的後處理器,用於自定義一些請求處理的擴展點;
MockMultipartHttpServletRequestBuilder繼承自MockHttpServletRequestBuilder,又提供了以下API:
MockMultipartHttpServletRequestBuilder file(String name, byte[] content)/MockMultipartHttpServletRequestBuilder file(MockMultipartFile file):指定要上傳的文件;
ResultActions
調用MockMvc.perform(RequestBuilder requestBuilder)後將獲得ResultActions,經過ResultActions完成以下三件事:
ResultActions andExpect(ResultMatcher matcher) :添加驗證斷言來判斷執行請求後的結果是不是預期的;
ResultActions andDo(ResultHandler handler) :添加結果處理器,用於對驗證成功後執行的動做,如輸出下請求/結果信息用於調試;
MvcResult andReturn() :返回驗證成功後的MvcResult;用於自定義驗證/下一步的異步處理;
ResultMatcher/MockMvcResultMatchers
ResultMatcher用來匹配執行完請求後的結果驗證,其就一個match(MvcResult result)斷言方法,若是匹配失敗將拋出相應的異常;spring mvc測試框架提供了不少***ResultMatchers來知足測試需求。注意這些***ResultMatchers並非ResultMatcher的子類,而是返回ResultMatcher實例的。Spring mvc測試框架爲了測試方便提供了MockMvcResultMatchers靜態工廠方法方便操做;具體的API以下:
HandlerResultMatchers handler():請求的Handler驗證器,好比驗證處理器類型/方法名;此處的Handler其實就是處理請求的控制器;
RequestResultMatchers request():獲得RequestResultMatchers驗證器;
ModelResultMatchers model():獲得模型驗證器;
ViewResultMatchers view():獲得視圖驗證器;
FlashAttributeResultMatchers flash():獲得Flash屬性驗證;
StatusResultMatchers status():獲得響應狀態驗證器;
HeaderResultMatchers header():獲得響應Header驗證器;
CookieResultMatchers cookie():獲得響應Cookie驗證器;
ContentResultMatchers content():獲得響應內容驗證器;
JsonPathResultMatchers jsonPath(String expression, Object ... args)/ResultMatcher jsonPath(String expression, Matcher<T> matcher):獲得Json表達式驗證器;
XpathResultMatchers xpath(String expression, Object... args)/XpathResultMatchers xpath(String expression, Map<String, String> namespaces, Object... args):獲得Xpath表達式驗證器;
ResultMatcher forwardedUrl(final String expectedUrl):驗證處理完請求後轉發的url(絕對匹配);
ResultMatcher forwardedUrlPattern(final String urlPattern):驗證處理完請求後轉發的url(Ant風格模式匹配,@since spring4);
ResultMatcher redirectedUrl(final String expectedUrl):驗證處理完請求後重定向的url(絕對匹配);
ResultMatcher redirectedUrlPattern(final String expectedUrl):驗證處理完請求後重定向的url(Ant風格模式匹配,@since spring4);
獲得相應的***ResultMatchers後,接着再調用其相應的API獲得ResultMatcher,如ModelResultMatchers.attributeExists(final String... names)判斷Model屬性是否存在。具體請查看相應的API。再次就不一一列舉了。
ResultHandler/MockMvcResultHandlers
ResultHandler用於對處理的結果進行相應處理的,好比輸出整個請求/響應等信息方便調試,Spring mvc測試框架提供了MockMvcResultHandlers靜態工廠方法,該工廠提供了ResultHandler print()返回一個輸出MvcResult詳細信息到控制檯的ResultHandler實現。
MvcResult
即執行完控制器後獲得的整個結果,並不只僅是返回值,其包含了測試時須要的全部信息,如:
MockHttpServletRequest getRequest():獲得執行的請求;
MockHttpServletResponse getResponse():獲得執行後的響應;
Object getHandler():獲得執行的處理器,通常就是控制器;
HandlerInterceptor[] getInterceptors():獲得對處理器進行攔截的攔截器;
ModelAndView getModelAndView():獲得執行後的ModelAndView;
Exception getResolvedException():獲得HandlerExceptionResolver解析後的異常;
FlashMap getFlashMap():獲得FlashMap;
Object getAsyncResult()/Object getAsyncResult(long timeout):獲得異步執行的結果;
1.5 測試示例
測試普通控制器
Java代碼
- //測試普通控制器
- mockMvc.perform(get("/user/{id}", 1)) //執行請求
- .andExpect(model().attributeExists("user")) //驗證存儲模型數據
- .andExpect(view().name("user/view")) //驗證viewName
- .andExpect(forwardedUrl("/WEB-INF/jsp/user/view.jsp"))//驗證視圖渲染時forward到的jsp
- .andExpect(status().isOk())//驗證狀態碼
- .andDo(print()); //輸出MvcResult到控制檯
測試普通控制器,可是URL錯誤,即404
Java代碼
- //找不到控制器,404測試
- MvcResult result = mockMvc.perform(get("/user2/{id}", 1)) //執行請求
- .andDo(print())
- .andExpect(status().isNotFound()) //驗證控制器不存在
- .andReturn();
- Assert.assertNull(result.getModelAndView()); //自定義斷言
獲得MvcResult自定義驗證
Java代碼
- MvcResult result = mockMvc.perform(get("/user/{id}", 1))//執行請求
- .andReturn(); //返回MvcResult
- Assert.assertNotNull(result.getModelAndView().getModel().get("user")); //自定義斷言
驗證請求參數綁定到模型數據及Flash屬性
Java代碼
- mockMvc.perform(post("/user").param("name", "zhang")) //執行傳遞參數的POST請求(也能夠post("/user?name=zhang"))
- .andExpect(handler().handlerType(UserController.class)) //驗證執行的控制器類型
- .andExpect(handler().methodName("create")) //驗證執行的控制器方法名
- .andExpect(model().hasNoErrors()) //驗證頁面沒有錯誤
- .andExpect(flash().attributeExists("success")) //驗證存在flash屬性
- .andExpect(view().name("redirect:/user")); //驗證視圖
驗證請求參數驗證失敗出錯
Java代碼
- mockMvc.perform(post("/user").param("name", "admin")) //執行請求
- .andExpect(model().hasErrors()) //驗證模型有錯誤
- .andExpect(model().attributeDoesNotExist("name")) //驗證存在錯誤的屬性
- .andExpect(view().name("showCreateForm")); //驗證視圖
文件上傳
Java代碼
- //文件上傳
- byte[] bytes = new byte[] {1, 2};
- mockMvc.perform(fileUpload("/user/{id}/icon", 1L).file("icon", bytes)) //執行文件上傳
- .andExpect(model().attribute("icon", bytes)) //驗證屬性相等性
- .andExpect(view().name("success")); //驗證視圖
JSON請求/響應驗證
測試時須要安裝jackson Json和JsonPath依賴:
Java代碼
- <dependency>
- <groupId>com.fasterxml.jackson.core</groupId>
- <artifactId>jackson-databind</artifactId>
- <version>${jackson2.version}</version>
- </dependency>
-
- <dependency>
- <groupId>com.jayway.jsonpath</groupId>
- <artifactId>json-path</artifactId>
- <version>${jsonpath.version}</version>
- <scope>test</scope>
- </dependency>
版本:<jsonpath.version>0.9.0</jsonpath.version>、<jackson2.version>2.2.3</jackson2.version> Java代碼
- String requestBody = "{\"id\":1, \"name\":\"zhang\"}";
- mockMvc.perform(post("/user")
- .contentType(MediaType.APPLICATION_JSON).content(requestBody)
- .accept(MediaType.APPLICATION_JSON)) //執行請求
- .andExpect(content().contentType(MediaType.APPLICATION_JSON)) //驗證響應contentType
- .andExpect(jsonPath("$.id").value(1)); //使用Json path驗證JSON 請參考http://goessner.net/articles/JsonPath/
-
- String errorBody = "{id:1, name:zhang}";
- MvcResult result = mockMvc.perform(post("/user")
- .contentType(MediaType.APPLICATION_JSON).content(errorBody)
- .accept(MediaType.APPLICATION_JSON)) //執行請求
- .andExpect(status().isBadRequest()) //400錯誤請求
- .andReturn();
-
- Assert.assertTrue(HttpMessageNotReadableException.class.isAssignableFrom(result.getResolvedException().getClass()));//錯誤的請求內容體
XML請求/響應驗證
測試時須要安裝spring oxm和xstream依賴:
Java代碼
- <dependency>
- <groupId>com.thoughtworks.xstream</groupId>
- <artifactId>xstream</artifactId>
- <version>${xsream.version}</version>
- <scope>test</scope>
- </dependency>
-
- <dependency>
- <groupId>org.springframework</groupId>
- <artifactId>spring-oxm</artifactId>
- <version>${spring.version}</version>
- <scope>test</scope>
- </dependency>
版本:<xstream.version>1.4.4</xstream.version>Java代碼
- //XML請求/響應
- String requestBody = "<user><id>1</id><name>zhang</name></user>";
- mockMvc.perform(post("/user")
- .contentType(MediaType.APPLICATION_XML).content(requestBody)
- .accept(MediaType.APPLICATION_XML)) //執行請求
- .andDo(print())
- .andExpect(content().contentType(MediaType.APPLICATION_XML)) //驗證響應contentType
- .andExpect(xpath("/user/id/text()").string("1")); //使用XPath表達式驗證XML 請參考http://www.w3school.com.cn/xpath/
-
- String errorBody = "<user><id>1</id><name>zhang</name>";
- MvcResult result = mockMvc.perform(post("/user")
- .contentType(MediaType.APPLICATION_XML).content(errorBody)
- .accept(MediaType.APPLICATION_XML)) //執行請求
- .andExpect(status().isBadRequest()) //400錯誤請求
- .andReturn();
-
- Assert.assertTrue(HttpMessageNotReadableException.class.isAssignableFrom(result.getResolvedException().getClass()));//錯誤的請求內容體
異常處理
Java代碼
- //異常處理
- MvcResult result = mockMvc.perform(get("/user/exception")) //執行請求
- .andExpect(status().isInternalServerError()) //驗證服務器內部錯誤
- .andReturn();
-
- Assert.assertTrue(IllegalArgumentException.class.isAssignableFrom(result.getResolvedException().getClass()));
靜態資源
Java代碼
- //靜態資源
- mockMvc.perform(get("/static/app.js")) //執行請求
- .andExpect(status().isOk()) //驗證狀態碼200
- .andExpect(content().string(CoreMatchers.containsString("var")));//驗證渲染後的視圖內容包含var
-
- mockMvc.perform(get("/static/app1.js")) //執行請求
- .andExpect(status().isNotFound()); //驗證狀態碼404
異步測試
Java代碼
- //Callable
- MvcResult result = mockMvc.perform(get("/user/async1?id=1&name=zhang")) //執行請求
- .andExpect(request().asyncStarted())
- .andExpect(request().asyncResult(CoreMatchers.instanceOf(User.class))) //默認會等10秒超時
- .andReturn();
-
- mockMvc.perform(asyncDispatch(result))
- .andExpect(status().isOk())
- .andExpect(content().contentType(MediaType.APPLICATION_JSON))
- .andExpect(jsonPath("$.id").value(1));
Java代碼
- //DeferredResult
- result = mockMvc.perform(get("/user/async2?id=1&name=zhang")) //執行請求
- .andExpect(request().asyncStarted())
- .andExpect(request().asyncResult(CoreMatchers.instanceOf(User.class))) //默認會等10秒超時
- .andReturn();
-
- mockMvc.perform(asyncDispatch(result))
- .andExpect(status().isOk())
- .andExpect(content().contentType(MediaType.APPLICATION_JSON))
- .andExpect(jsonPath("$.id").value(1));
此處請在第一次請求時加上 andExpect(request().asyncResult(CoreMatchers.instanceOf(User.class)))這樣會等待結果返回/超時,無須本身設置線程等待了;此處注意request().asyncResult必定是在第一次請求發出;而後第二次經過asyncDispatch進行異步請求。
添加自定義過濾器
Java代碼
- mockMvc = webAppContextSetup(wac).addFilter(new MyFilter(), "/*").build();
- mockMvc.perform(get("/user/1"))
- .andExpect(request().attribute("filter", true));
全局配置
Java代碼
- mockMvc = webAppContextSetup(wac)
- .defaultRequest(get("/user/1").requestAttr("default", true)) //默認請求 若是其是Mergeable類型的,會自動合併的哦mockMvc.perform中的RequestBuilder
- .alwaysDo(print()) //默認每次執行請求後都作的動做
- .alwaysExpect(request().attribute("default", true)) //默認每次執行後進行驗證的斷言
- .build();
-
- mockMvc.perform(get("/user/1"))
- .andExpect(model().attributeExists("user"));
以上代碼請參考個人github。更多參考示例請前往Spring github。
只要記住測試步驟,按照步驟操做,整個測試過程是很是容易理解的:
一、準備測試環境
二、經過MockMvc執行請求
3.一、添加驗證斷言
3.二、添加結果處理器
3.三、獲得MvcResult進行自定義斷言/進行下一步的異步請求
四、卸載測試環境