Тестовая среда Spring MVC - конечная точка JSON возвращает 406 только в тесте

Я создал работающее веб-приложение Spring MVC с использованием Hibernate и JPA с одной конечной точкой, которая успешно возвращает JSON только при вызове через веб-браузер.

Я также прошел модульные тесты для классов, в частности для основного контроллера, который отображает конечную точку.

Поэтому моим следующим шагом было написать несколько поведенческих тестов с использованием среды тестирования Spring MVC для тестирования конечной точки JSON со всеми ее рабочими частями (конфигурацией, контроллером, сервисом, моделью и репозиторием).

Я успешно настроил тест, который использует конфигурацию моего приложения, вводит/автопроводит все вместе, а затем делает запрос «GET» к моей конечной точке. Я могу отлаживать вызовы, отслеживая действия от конечной точки контроллера через службу до репозитория, успешно получая данные и возвращая правильный объект модели. Однако затем контроллер возвращает ответ 406 not accept, и тест завершается неудачно.

Я провел 2 дня, работая над этой проблемой, и прочитал почти все сообщения в блогах и ТАК ответ, который я могу принять без каких-либо предложений.

Вот мой контроллер:

@Controller
public class DataController {

    @Autowired
    private IService service;

    @RequestMapping(
            value = "/data",
            method = RequestMethod.GET,
            produces = "application/json"
    )
    public @ResponseBody
    ModelData getDataByCode(@RequestParam String code) {
        return service.getDataCode(code);
    }

}

Услуга:

@Service("dataService")
public class DataService implements IService {

    @Autowired
    private IDataRepository dataRepository;

    @Transactional
    public ModelData getDataByCode(String code){
        return dataRepository.getDataByCode(code);
    }
}

Репозиторий:

@Repository("dataRepository")
public class DataRepository implements IDataRepository {

    @PersistenceContext
    private EntityManager entityManager;

    public ModelData getDataByCode(String code) throws IllegalStateException, PersistenceException {
        String selectJPAQueryBase = "Select data from ModelData data where data.code = %s";
        Query query = entityManager.createQuery(String.format(selectJPAQueryBase, code));
        return (ModelData) query.getSingleResult();
    }

}

Зависимости, относящиеся к основным функциям:

        <dependency>
            <groupId>org.springframework</groupId>
            <artifactId>spring-webmvc</artifactId>
            <version>3.2.14.RELEASE</version>
        </dependency>
        <dependency>
            <groupId>javax.servlet</groupId>
            <artifactId>servlet-api</artifactId>
            <version>2.5</version>
            <scope>provided</scope>
        </dependency>
        <dependency>
            <groupId>javax.servlet</groupId>
            <artifactId>jstl</artifactId>
            <version>1.2</version>
            <scope>provided</scope>
        </dependency>
        <dependency>
            <groupId>com.fasterxml.jackson.core</groupId>
            <artifactId>jackson-core</artifactId>
            <version>2.6.2</version>
        </dependency>
        <dependency>
            <groupId>com.fasterxml.jackson.core</groupId>
            <artifactId>jackson-databind</artifactId>
            <version>2.6.2</version>
        </dependency>
        <dependency>
            <groupId>com.fasterxml.jackson.core</groupId>
            <artifactId>jackson-annotations</artifactId>
            <version>2.6.2</version>
        </dependency>
        <dependency>
            <groupId>org.springframework</groupId>
            <artifactId>spring-oxm</artifactId>
            <version>3.2.14.RELEASE</version>
        </dependency>
        <dependency>
            <groupId>org.hibernate</groupId>
            <artifactId>hibernate-validator</artifactId>
            <version>4.2.0.Final</version>
        </dependency>
        <dependency>
            <groupId>org.hibernate</groupId>
            <artifactId>hibernate-entitymanager</artifactId>
            <version>5.0.1.Final</version>
        </dependency>
        <dependency>
            <groupId>javax.transaction</groupId>
            <artifactId>jta</artifactId>
            <version>1.1</version>
        </dependency>
        <dependency>
            <groupId>org.springframework</groupId>
            <artifactId>spring-jdbc</artifactId>
            <version>3.2.14.RELEASE</version>
        </dependency>
        <dependency>
            <groupId>org.springframework</groupId>
            <artifactId>spring-orm</artifactId>
            <version>3.2.14.RELEASE</version>
        </dependency>
        <dependency>
            <groupId>net.sourceforge.jtds</groupId>
            <artifactId>jtds</artifactId>
            <version>1.3.1</version>
        </dependency>
        <dependency>
            <groupId>junit</groupId>
            <artifactId>junit</artifactId>
            <version>4.12</version>
            <scope>test</scope>
        </dependency>
        <dependency>
            <groupId>info.cukes</groupId>
            <artifactId>cucumber-java</artifactId>
            <version>1.2.4</version>
            <scope>test</scope>
        </dependency>
        <dependency>
            <groupId>info.cukes</groupId>
            <artifactId>cucumber-junit</artifactId>
            <version>1.2.4</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.springframework</groupId>
            <artifactId>spring-test</artifactId>
            <version>3.2.14.RELEASE</version>
            <scope>test</scope>
        </dependency>
        <dependency>
            <groupId>com.jayway.jsonpath</groupId>
            <artifactId>json-path</artifactId>
            <version>0.8.1</version>
            <scope>test</scope>
        </dependency>

Контекст JPA:

<?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:tx="http://www.springframework.org/schema/tx"
       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/tx http://www.springframework.org/schema/tx/spring-tx-3.2.xsd
            http://www.springframework.org/schema/context http://www.springframework.org/schema/context/spring-context-3.2.xsd">

    <context:annotation-config />

    <!-- where spring will start to look for classes that are annotated, allow this to find all annotated classes -->
    <context:component-scan base-package="com.app"/>

    <bean class="org.springframework.orm.jpa.support.PersistenceAnnotationBeanPostProcessor" />

    <bean id="entityManagerFactory" class="org.springframework.orm.jpa.LocalContainerEntityManagerFactoryBean">
        <property name="persistenceUnitName" value="punit"/>
        <property name="dataSource" ref="dataSource"/>
        <property name="jpaVendorAdapter">
            <bean class="org.springframework.orm.jpa.vendor.HibernateJpaVendorAdapter">
                <property name="showSql" value="true"/>
            </bean>
        </property>
        <property name="jpaPropertyMap">
            <map>
                <entry key="hibernate.dialect" value="org.hibernate.dialect.SQLServer2008Dialect"/>
                <entry key="hibernate.hbm2ddl.auto" value="validate"/>
                <entry key="hibernate.format_sql" value="true"/>
            </map>
        </property>
    </bean>

    <bean id="transactionManager" class="org.springframework.orm.jpa.JpaTransactionManager">
        <property name="entityManagerFactory" ref="entityManagerFactory"/>
    </bean>

    <tx:annotation-driven transaction-manager="transactionManager"/>

    <bean id="dataSource" class="org.springframework.jdbc.datasource.DriverManagerDataSource">
        <property name="driverClassName" value="net.sourceforge.jtds.jdbc.Driver"/>
        <property name="url" value="jdbc:jtds:sqlserver://0.0.0.0:1433/dbname"/>
        <property name="username" value="username"/>
        <property name="password" value="password"/>
    </bean>

</beans>

Веб.xml:

<?xml version="1.0" encoding="UTF-8"?>
<web-app version="2.5" xmlns="http://java.sun.com/xml/ns/javaee"
    xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
    xsi:schemaLocation="http://java.sun.com/xml/ns/javaee http://java.sun.com/xml/ns/javaee/web-app_2_5.xsd">

    <context-param>
        <param-name>contextConfigLocation</param-name>
        <param-value>classpath:/jpaContext.xml</param-value>
    </context-param>

    <listener>
        <listener-class>org.springframework.web.context.ContextLoaderListener</listener-class>
    </listener>

    <servlet>
        <servlet-name>dataByCode</servlet-name>
        <servlet-class>org.springframework.web.servlet.DispatcherServlet</servlet-class>
        <init-param>
            <param-name>contextConfigLocation</param-name>
            <param-value>/WEB-INF/config/servlet-config.xml</param-value>
        </init-param>
    </servlet>

    <servlet-mapping>
        <servlet-name>dataByCode</servlet-name>
        <url-pattern>/data</url-pattern>
    </servlet-mapping>

</web-app>

Конфигурация сервлета:

<?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:mvc="http://www.springframework.org/schema/mvc"
       xmlns:context="http://www.springframework.org/schema/context"
       xsi:schemaLocation="http://www.springframework.org/schema/mvc http://www.springframework.org/schema/mvc/spring-mvc-3.2.xsd
        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-3.2.xsd">

    <!-- allows POJO objects to just use annotations for simply hooking into spring framework -->
    <mvc:annotation-driven/>

    <!-- where spring will start to look annotated classes, keep this to controller level -->
    <context:component-scan base-package="com.app"/>

    <!-- load properties files and similar -->
    <bean id="messageSource" class="org.springframework.context.support.ResourceBundleMessageSource">
        <property name="basename" value="properties"/>
    </bean>

    <!-- basic resolver to handle jsp pages, usually linked to returning html -->
    <bean class="org.springframework.web.servlet.view.InternalResourceViewResolver">
        <property name="prefix" value="/WEB-INF/jsp/"/>
        <property name="suffix" value=".jsp"/>
        <property name="order" value="2"/>
    </bean>

</beans>

Неудачный тест:

@RunWith(SpringJUnit4ClassRunner.class)
@WebAppConfiguration
@ContextConfiguration("classpath:jpaContext.xml")
public class GetDataCodeIntTest {

    @Autowired
    private WebApplicationContext webApplicationContext;

    private MockMvc mockMvc;

    @Before
    public void setup() {
        this.mockMvc = MockMvcBuilders.webAppContextSetup(this.webApplicationContext).build();
    }

    @Test
    public void getDataByCode() throws Exception {
        String code = "1234";
        this.mockMvc.perform(MockMvcRequestBuilders.get("/data").param("code", code))
                .andExpect(MockMvcResultMatchers.status().isOk())
                .andExpect(MockMvcResultMatchers.content().contentType(MediaType.APPLICATION_JSON))
                .andExpect(MockMvcResultMatchers.jsonPath("$.code").value(code));
    }

}

Ключевым моментом здесь является то, что приложение отлично работает через браузер, но не работает с тестовой средой Spring MVC в момент, когда контроллер возвращает ответ.


ОБНОВИТЬ:

Решение состояло в том, чтобы внедрить мой servlet-config.xml вместе с jpaContext в @ContextConfiguration, после этого тест прошел отлично.

Интересно, что я не мог просто загрузить servlet-config.xml из пути к классам, и я не хотел поддерживать конкретную тестовую версию, поэтому я переместил его в папку с ресурсами. Было ли это хорошей идеей, я не решил, но теперь это работает из пути к классам.


person Jeremy    schedule 07.12.2015    source источник
comment
Попробуйте установить @Consumes, если вы устанавливаете accept, также проверьте зависимости как здесь описано   -  person ernestk    schedule 07.12.2015
comment
Где мне поставить @Consumes, пожалуйста?   -  person Jeremy    schedule 07.12.2015
comment
Я безуспешно изменил версии в соответствии со связанным ответом, все еще 406 в тесте. Плюс, если честно, весь код работает с текущими зависимостями, он просто терпит неудачу с тестом Spring MVC Test Framework.   -  person Jeremy    schedule 07.12.2015
comment
извините, это было плохое предложение   -  person ernestk    schedule 07.12.2015


Ответы (3)


Попробуйте добавить servletConfig.xml к @ContextConfiguration в тестовом классе:

@ContextConfiguration({"classpath:jpaContext.xml", "classpath:path/to/servlet-context.xml"})

Важным битом здесь является <mvc:annotation-driven/>.

Также обратите внимание, что вы можете немного упростить свой контроллер и прояснить его назначение, аннотировав его @RestController вместо @Controller.

person whistling_marmot    schedule 07.12.2015

Попробуйте добавить свой spring-context.xml для тестирования через @ContextConfiguration (например: @ContextConfiguration("classpath:spring-context.xml")), попробовал ваш код с этой опцией, это сработало.

person ernestk    schedule 07.12.2015

Я думаю, что ваша проблема заключается в том, что вы неправильно понимаете свой заголовок «Content-type», и результат создается вашим контроллером REST.

На самом деле вы положили

MockMvcResultMatchers.content (). ContentType (MediaType.APPLICATION_JSON)

означает, что вы отправляете JSON на свой контроллер. Принимая во внимание, что ваш контроллер пытается получить параметр в вашем запросе, а не JSON:

@RequestParam Строковый код

Вероятно, причина, по которой у вас есть этот 406 (что обычно означает значение по умолчанию в заголовке вашего запроса).

person Julien Alary    schedule 07.12.2015
comment
До этого он терпит неудачу в .andExpect(MockMvcResultMatchers.status().isOk()). Кроме того, я пробовал с указанием типа контента и без него в тесте, и он все равно терпит неудачу с ошибкой 406. - person Jeremy; 07.12.2015
comment
Я почти уверен, что .andExpect(MockMvcResultMatchers.content().contentType(MediaType.APPLICATION_JSON)) проверяет, что данные, отправляемые обратно из контроллера, имеют тип JSON, и это именно то, что я хочу проверить. или, по крайней мере, заголовок в запросе на возврат в любом случае имеет тип содержимого «application/json». - person Jeremy; 07.12.2015
comment
Хотя он указывает на результат, я думаю, он имеет в виду заголовок принятия, а не проверку результата. Удалите часть accept, так как это не то, что отправит браузер (если вы не используете JS и сами явно не устанавливаете тип контента). - person M. Deinum; 07.12.2015
comment
Да, я согласен, это было там только потому, что я экспериментировал с конкретной настройкой типа содержимого заголовка для запроса. я удалил его и тот же результат, тест 406 не проходит. Я отрегулирую детали, чтобы отразить это. - person Jeremy; 07.12.2015