Spring MVC (неподдерживаемый тип носителя)

#spring #spring-mvc

#spring #spring-mvc

Вопрос:

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

 <mvc:annotation-driven />
  

он должен автоматически инициализировать MappingJackson2HttpMessageConverter то, что мне требуется для передачи моего объекта контроллеру в форме JSON, и у меня есть эти библиотеки в моем пути к классам в виде зависимостей maven.

 <dependency>
  <groupId>com.fasterxml.jackson.core</groupId>
  <artifactId>jackson-annotations</artifactId>
  <version>2.3.2</version>
</dependency>

<dependency>
  <groupId>com.fasterxml.jackson.core</groupId>
  <artifactId>jackson-core</artifactId>
  <version>2.3.2</version>
</dependency>

<dependency>
  <groupId>com.fasterxml.jackson.core</groupId>
  <artifactId>jackson-databind</artifactId>
  <version>2.3.2</version>
</dependency>
  

итак, теперь у меня есть пространство имен MVC, есть библиотеки message converter json в classpath, но все равно он выдает ошибку. это мой тестовый файл.

PersonControllerTest.java

 package com.prateekj.controllers;

import com.fasterxml.jackson.databind.ObjectMapper;
import com.prateekj.maker.PersonMaker;
import com.prateekj.model.Person;
import com.prateekj.services.PersonService;
import org.junit.Before;
import org.junit.Test;
import org.junit.runner.RunWith;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.http.MediaType;
import org.springframework.test.context.ContextConfiguration;
import org.springframework.test.context.junit4.SpringJUnit4ClassRunner;
import org.springframework.test.context.web.WebAppConfiguration;
import org.springframework.test.web.servlet.MockMvc;
import org.springframework.test.web.servlet.setup.MockMvcBuilders;
import org.springframework.web.context.WebApplicationContext;

import static com.natpryce.makeiteasy.MakeItEasy.a;
import static com.natpryce.makeiteasy.MakeItEasy.make;
import static com.natpryce.makeiteasy.MakeItEasy.with;
import static org.mockito.Mockito.verify;
import static org.springframework.test.web.servlet.request.MockMvcRequestBuilders.put;
import static org.springframework.test.web.servlet.result.MockMvcResultMatchers.status;

@RunWith(SpringJUnit4ClassRunner.class)
@WebAppConfiguration
@ContextConfiguration("classpath:/configuration/Beans.xml")
public class PersonControllerTest {

  @Autowired
  private WebApplicationContext wac;

  private MockMvc mockMvc;

  @Autowired
  private PersonService personService;

  private Person person;

  private Integer DEFAULT_ID = 2;

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

  @Test
  public void shouldAddTheUser() throws Exception {
    person = make(a(PersonMaker.Person, with(PersonMaker.id, (Integer)null)));

    mockMvc.perform(put("/users/add").contentType(MediaType.APPLICATION_JSON).content(new ObjectMapper().writeValueAsString(person)))
        .andExpect(status().isCreated());

    verify(personService).savePerson(person);
  }
}
  

Beans.xml

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

  <jpa:repositories base-package="com.prateekj.repositories"/>
  <context:component-scan base-package="com.prateekj.services"/>
  <context:component-scan base-package="com.prateekj.controllers"/>

  <bean id="configurationEncryptor" class="org.jasypt.encryption.pbe.StandardPBEStringEncryptor">
    <property name="config">
      <bean class="org.jasypt.encryption.pbe.config.SimpleStringPBEConfig">
        <property name="algorithm" value="PBEWithMD5AndDES"/>
        <property name="password" value="password"/>
      </bean>
    </property>
  </bean>

  <bean id="propertyConfigurer"
        class="org.jasypt.spring31.properties.EncryptablePropertyPlaceholderConfigurer">
    <constructor-arg ref="configurationEncryptor"/>
    <property name="locations">
      <list>
        <value>classpath:properties/persistence.properties</value>
      </list>
    </property>
  </bean>


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

  <bean id="entityManagerFactory" class="org.springframework.orm.jpa.LocalContainerEntityManagerFactoryBean">
    <property name="dataSource" ref="dataSource"/>
    <property name="jpaVendorAdapter" ref="jpaVendorAdapter"/>
    <property name="persistenceUnitName" value="mysql-core"/>
  </bean>

  <bean id="jpaVendorAdapter" class="org.springframework.orm.jpa.vendor.HibernateJpaVendorAdapter">
    <property name="database" value="MYSQL"/>
    <property name="databasePlatform" value="org.hibernate.dialect.MySQL5Dialect"/>
    <property name="showSql" value="false"/>
  </bean>

  <bean id="dataSource" class="com.mchange.v2.c3p0.ComboPooledDataSource" destroy-method="close">
    <property name="driverClass" value="${com.prateekj.jdbc.driver}"/>
    <property name="jdbcUrl" value="${com.prateekj.jdbc.url}"/>
    <property name="user" value="${com.prateekj.jdbc.user}"/>
    <property name="password" value="${com.prateekj.jdbc.password}"/>
  </bean>


</beans>
  

web.xml

 <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">



  <servlet>
    <servlet-name>toDo</servlet-name>
    <servlet-class>org.springframework.web.servlet.DispatcherServlet</servlet-class>
    <init-param>
      <param-name>contextConfigLocation</param-name>
      <param-value>classpath:configuration/mvc-dispatcher-config.xml</param-value>
    </init-param>
  </servlet>

  <servlet-mapping>
    <servlet-name>toDo</servlet-name>
    <url-pattern>/</url-pattern>
  </servlet-mapping>


</web-app>
  

mvc-dispatcher-config.xml

 <?xml version="1.0" encoding="UTF-8"?>
<beans xmlns="http://www.springframework.org/schema/beans"
       xmlns:mvc="http://www.springframework.org/schema/mvc"
       xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
       xsi:schemaLocation="
        http://www.springframework.org/schema/beans
        http://www.springframework.org/schema/beans/spring-beans.xsd
        http://www.springframework.org/schema/mvc
        http://www.springframework.org/schema/mvc/spring-mvc.xsd">

  <mvc:annotation-driven />

</beans>
  

PersonController.java

 package com.prateekj.controllers;

import com.prateekj.model.Person;
import com.prateekj.services.PersonService;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.http.HttpStatus;
import org.springframework.http.ResponseEntity;
import org.springframework.stereotype.Controller;
import org.springframework.web.bind.annotation.RequestBody;
import org.springframework.web.bind.annotation.RequestMapping;
import org.springframework.web.bind.annotation.RequestMethod;

@Controller
@RequestMapping(value = "/users")
public class PersonController {

  private PersonService personService;

  @Autowired
  public PersonController(PersonService personService){
    this.personService = personService;
  }

  @RequestMapping(value = "/add", method = RequestMethod.PUT)
  public ResponseEntity<Void> addUser(@RequestBody Person person){
    personService.savePerson(person);
    return new ResponseEntity<Void>(HttpStatus.CREATED);
  }
}
  

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

Комментарии:

1. Вы показываете довольно полное приложение, включая преобразование json. Работал ли он до последнего изменения, и если да, то что это была за модификация? Если нет, вы должны попытаться сократить его до простого кода, который показывает проблему.

2. @SergeBallesta, я начал писать код с уровня репозитория и до уровня обслуживания, все прошло хорошо, и когда я перешел к контроллеру, мне пришлось реализовать эти файлы сервлетов диспетчера, веб-дескриптор и т. Д., Так что вы можете сказать, что это был большой шаг, предпринятый мной для реализации контроллераи подключение их к уровню обслуживания. итак, вы спросили о последней модификации. Следовательно, для реализации контроллера была последняя модификация, сделанная мной.

3. @NimChimpsky теперь я обновил его с помощью кода контроллера,

4. Просто добавьте свой mvc-dispatcher-config.xml to @ContextConfiguration в свои тесты.

Ответ №1:

Вам не хватает нескольких шагов в ваших файлах конфигурации.

В web.xml , вы никогда не запускаете контекст корневого приложения. Вам нужно что-то вроде :

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

и поскольку вы не используете значение по умолчанию /WEB-INF/applicationContext.xml

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

Это не имеет ничего общего с вашей текущей ошибкой, но вскоре это должно привести к другим проблемам.

Вы хотите использовать JSON и правильно объявляете его в своем pom.xml . Но вы забыли сказать spring framework, что вам нужен конвертер JSON. => причина вашего Unsupported Media Type . Вы должны добавить что-то подобное в свой web-dispatcher-config.xml

 <!-- Configure to plugin JSON as request and response in method handler -->
<beans:bean class="org.springframework.web.servlet.mvc.method.annotation.RequestMappingHandlerAdapter">
    <beans:property name="messageConverters">
        <beans:list>
            <beans:ref bean="jsonMessageConverter"/>
        </beans:list>
    </beans:property>
</beans:bean>

<!-- Configure bean to convert JSON to POJO and vice versa -->
<beans:bean id="jsonMessageConverter"
    class="org.springframework.http.converter.json.MappingJackson2HttpMessageConverter">
</beans:bean>   
  

Если вы новичок в Spring MVC, я настоятельно рекомендую вам следовать руководству. Я просто погуглил и быстро нашел 2 (в моем текущем сообщении используются выдержки из second) :