MockMvc для тестирования веб-службы SpringMVC Rest с использованием JUnit, Mockito и Hamcrest

Я использую Spring MVC для создания Restful Web Services...

Вот мой pom.xml:

<dependency>
    <groupId>org.springframework</groupId>
    <artifactId>spring-core</artifactId>
    <version>4.0.3.RELEASE</version>
</dependency>

<dependency>
    <groupId>org.springframework</groupId>
    <artifactId>spring-test</artifactId>
    <version>4.0.3.RELEASE</version>
    <scope>test</scope>
</dependency>

<dependency>
    <groupId>org.springframework</groupId>
    <artifactId>spring-web</artifactId>
    <version>4.0.3.RELEASE</version>
</dependency>

<dependency>
    <groupId>org.springframework</groupId>
    <artifactId>spring-webmvc</artifactId>
    <version>4.0.3.RELEASE</version>
</dependency>

<dependency>
    <groupId>junit</groupId>
    <artifactId>junit</artifactId>
    <version>4.12</version>
    <scope>test</scope>
</dependency>

<dependency>
    <groupId>org.hamcrest</groupId>
    <artifactId>hamcrest-all</artifactId>
    <version>1.3</version>
    <scope>test</scope>
</dependency>

<dependency>
    <groupId>org.mockito</groupId>
    <artifactId>mockito-core</artifactId>
    <version>1.10.19</version>
    <scope>test</scope>
</dependency>

<dependency>
    <groupId>com.jayway.jsonpath</groupId>
    <artifactId>json-path-assert</artifactId>
    <version>0.9.1</version>
    <scope>test</scope>
</dependency>

ВЕБ-INF/web.xml:

<?xml version="1.0" encoding="UTF-8"?>
<web-app id="WebApp_ID" version="2.4"
    xmlns="http://java.sun.com/xml/ns/j2ee" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
    xsi:schemaLocation="http://java.sun.com/xml/ns/j2ee http://java.sun.com/xml/ns/j2ee/web-app_2_4.xsd">
    <display-name>MyApp</display-name>
    <welcome-file-list>
        <welcome-file>index.html</welcome-file>
        <welcome-file>index.jsp</welcome-file>
    </welcome-file-list>
    <servlet>

        <servlet-name>mvc-dispatcher</servlet-name>
        <servlet-class>org.springframework.web.servlet.DispatcherServlet</servlet-class>
        <load-on-startup>1</load-on-startup>
    </servlet>
    <servlet-mapping>
        <servlet-name>mvc-dispatcher</servlet-name>
        <url-pattern>/</url-pattern>
    </servlet-mapping>

    <context-param>
        <param-name>contextConfigLocation</param-name>
        <param-value>/WEB-INF/mvc-dispatcher-servlet.xml</param-value>
    </context-param>
</web-app>

WEB-INF/mvc-dispatcher-servlet.xml:

<beans xmlns="http://www.springframework.org/schema/beans"
    <import resource="classpath:database_db.xml" />

    <context:component-scan base-package="com.myapp.rest" />
    <mvc:annotation-driven />

    <bean id="viewResolver"
        class="org.springframework.web.servlet.view.InternalResourceViewResolver">
        <property name="prefix">
            <value>/WEB-INF/pages/</value>
        </property>
        <property name="suffix">
            <value>.jsp</value>
        </property>
    </bean>
</beans>

src/main/resources/database_db.xml:

<beans xmlns="http://www.springframework.org/schema/beans">
    <bean id="dataSourceDB"  class="org.apache.commons.dbcp.BasicDataSource" destroy-method="close">
        <property name="driverClassName"><value>com.mysql.jdbc.Driver</value></property>
        <property name="url"><value>jdbc:mysql://localhost/mydatabase?zeroDateTimeBehavior=convertToNull</value></property>
        <property name="username"><value>root</value></property>
        <property name="password"><value></value></property>
    </bean> 
</beans>

Пользовательский контроллер:

@Controller
@RequestMapping("/v2")
public class UserController {

    private final UserDAO dao;

    @Autowired 
    public UserController(UserDAO dao) {
        this.dao = dao;
    }

    @RequestMapping(value = "users/appId", method = RequestMethod.GET)
    public @ResponseBody Object getUserDetails(@PathVariable String appId) {
        Object response = null;
        response = dao.getUser(appId);
        return response;
    }
}

UserDAO:

@Repository открытый класс UserDAO {

private JdbcTemplate jdbcTemplate;

@Autowired
public UserDAO(@Qualifier("dataSourceDB") DataSource dataSource) {
    this.jdbcTemplate = new JdbcTemplate(dataSource);
}   

// Various get and finder methods

}

источник/тест/Java:

@RunWith(SpringJUnit4ClassRunner.class)
@ContextConfiguration(locations = "classpath:**/mvc-dispatcher-servlet.xml")
@WebAppConfiguration
public class UserControllerTest {

    @Autowired
    private WebApplicationContext ctx;

    private MockMvc mockMvc;

    @InjectMocks
    private UserController userController;

    @Before
    public void setUp() {
        MockitoAnnotations.initMocks(this);
        mockMvc = MockMvcBuilders.standaloneSetup(userController).build();
    }

    @Test
    public void appUser() throws Exception {
        String appId = "1234FD57";

        mockMvc.perform(get("/v2/users/{appId}", appId)
                       .accept(MediaType.APPLICATION_JSON))
                       .andDo(print())
                       .andExpect(status().isOk());
    }
}

Вот результат:

-------------------------------------------------------
 T E S T S
-------------------------------------------------------
Running com.myapp.rest.controllers.UserControllerTest

MockHttpServletRequest:
         HTTP Method = GET
         Request URI = /v2/users/1234FD57
          Parameters = {}
             Headers = {Accept=[application/json]}

             Handler:
                Type = com.myapp.rest.controllers.UserController
              Method = public java.lang.Object com.myapp.rest.controllers.UserController.getUserDetails(java.lang.String)

               Async:
   Was async started = false
        Async result = null

  Resolved Exception:
                Type = null

        ModelAndView:
           View name = null
                View = null
               Model = null

            FlashMap:

MockHttpServletResponse:
              Status = 200
       Error message = null
             Headers = {}
        Content type = null
                Body = 
       Forwarded URL = null
      Redirected URL = null
             Cookies = []
Tests run: 1, Failures: 0, Errors: 0, Skipped: 0, Time elapsed: 0.751 sec

Тест выполняется, но тело не содержит JSON (обратите внимание, что оно пустое)...

Когда я изменяю содержимое своего тестового метода на:

mockMvc.perform(get("/v2/users/{appId}",appId)
                   .accept(MediaType.APPLICATION_JSON))
                   .andExpect(status().isOk())
                   .andExpect(jsonPath("$[1].userId", is("1234FD57")));

Получил следующую ошибку:

Tests run: 1, Failures: 0, Errors: 1, Skipped: 0, Time elapsed: 0.766 sec <<< FAILURE!

user(com.myapp.rest.controllers.UserControllerTest)  Time elapsed: 0.157 sec  <<< ERROR!
org.springframework.beans.factory.BeanCreationException: Error creating bean with name 'com.myapp.rest.controllers.UserControllerTest': Injection of autowired dependencies failed; nested exception is org.springframework.beans.factory.BeanCreationException: 

Could not autowire field: private com.myapp.rest.controllers.UserController 
    com.myapp.rest.controllers.UserControllerTest.userController; nested exception is org.springframework.beans.factory
    NoSuchBeanDefinitionException: No qualifying bean of type [com.myapp.rest.controllers.UserController] found for dependency: expected at least 1 bean which qualifies as autowire candidate for this dependency. Dependency annotations: {@org.mockito.InjectMocks(), @org.springframework.beans.factory.annotation.Autowired(required=true)}
    at org.springframework.beans.factory.annotation.AutowiredAnnotationBeanPostProcessor.postProcessPropertyValues(AutowiredAnnotationBeanPostProcessor.java:292)
    at org.springframework.beans.factory.support.AbstractAutowireCapableBeanFactory.populateBean(AbstractAutowireCapableBeanFactory.java:1185)
    at org.springframework.beans.factory.support.AbstractAutowireCapableBeanFactory.autowireBeanProperties

Это точно так же, как в заявлении print(), не показывающем ничего для «Body», нет JSON... Я очень близко, кто-нибудь знает, что я делаю?


person PacificNW_Lover    schedule 26.06.2017    source источник
comment
Не уверен, почему ваш запрос обрабатывается правильным обработчиком, но ваше сопоставление запросов отсутствует {} вокруг переменной пути. Должно быть @RequestMapping(value = "users/{appId}", method = RequestMethod.GET) Что такое @ResponseBodyObject? Разве не должно быть @ResponseBody?   -  person alfcope    schedule 26.06.2017
comment
Это была опечатка - отредактировал и исправил.   -  person PacificNW_Lover    schedule 26.06.2017


Ответы (2)


Поскольку вы работаете с Spring 4.0.3, вы должны использовать старый jsonpath 0.9.1, см. https://jira.spring.io/browse/SPR-12299

person Ralf Stuckert    schedule 26.06.2017
comment
Спасибо, я изменил его на 0.9.1 и теперь у меня новая ошибка... Скоро обновлю свой пост с новой ошибкой. - person PacificNW_Lover; 26.06.2017

Я не думаю, что вы инициализируете свой UserController в своем тесте. Вы также можете:

  1. Инициализируйте его в тесте самостоятельно. UserController userController = new UserController();
  2. @Autowire пользовательский контроллер. Но вам также необходимо включить конфигурацию с сканированием вашего компонента в @ContextConfiguration.
person Plog    schedule 26.06.2017
comment
Я только что попробовал то, что вы сказали... Просто обновил сообщение, чтобы показать вам, что произошло, используя ваше предложение. Теперь у меня проблема с проводкой Bean. - person PacificNW_Lover; 26.06.2017
comment
Ах, да, вам также нужно добавить свою конфигурацию, которая выполняет сканирование вашего компонента, в @ContextConfiguration. Или вы можете просто не подключать его автоматически, а просто выполнить UserController userController = new UserController() в тесте. - person Plog; 26.06.2017
comment
Сбой Autowire для контроллера... Что мне делать после создания экземпляра UserController? Можете ли вы привести пример сканирования компонентов в @ContextConfiguration (что там должно быть)? - person PacificNW_Lover; 28.06.2017
comment
Прочитав ваш код еще раз, я понимаю, что вы на самом деле сканируете компоненты в своем тесте, поскольку вы включаете mvc-dispatcher-servlet.xml (который содержит сканирование компонентов) в конфигурацию вашего контекста для теста, хотя по какой-то причине это не работает. Можете ли вы сказать мне, в каком пакете находится ваш UserController? Если он не находится в ветке com.myapp.rest, вам нужно добавить его в сканирование компонентов. Если это так, то моим единственным другим предложением было бы попробовать @ContextConfiguration(locations = classpath:mvc-dispatcher-servlet.xml) вместо того, что у вас есть сейчас. - person Plog; 28.06.2017
comment
Plog - Спасибо за ответ... Мой UserController находится под com.myapp.rest.controllers. Я добился большего прогресса (теперь он печатает Content-type, а также помещаю его под заголовки). Поскольку этот пост разветвлялся в другом направлении, и мне нужен контент JSON, см. этот новый пост, в котором есть более подробная информация (конфигурация xmls, имена пакетов), спасибо: stackoverflow. ком/вопросы/44793335/ - person PacificNW_Lover; 28.06.2017
comment
Без проблем. Хотя сейчас я понял, что то, что я говорил, было неправильным. Аннотации @InjectMocks достаточно для инициализации UserController. Извини! - person Plog; 28.06.2017