Spring MVC + JSON = 406 Не допускается
Я пытаюсь создать простой ответ JSON. Сейчас я получаю 406 недопустимую ошибку. Tomcat говорит: "Ресурс, идентифицированный этим запросом, способен генерировать ответы с характеристиками, неприемлемыми в соответствии с заголовками запроса" принять ". даже если мои заголовки Accept
Accept:text/html,application/xhtml+xml,application/xml;q=0.9,*/*;q=0.8
В tomcat/lib у меня есть все ящики Tomcat, Spring jars и jackson-all-1.9.0.jar. Я использую Spring 3.2.2 с Tomcat 7.
Я знаю, что этот вопрос обсуждался много раз, но ни один из решений не работает для меня.
web.xml
<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>Spring Web MVC Application</display-name>
<servlet>
<servlet-name>dispatcher</servlet-name>
<servlet-class>
org.springframework.web.servlet.DispatcherServlet
</servlet-class>
<load-on-startup>1</load-on-startup>
</servlet>
<servlet-mapping>
<servlet-name>dispatcher</servlet-name>
<url-pattern>*.htm</url-pattern>
</servlet-mapping>
</web-app>
диспетчер-servlet.xml
<beans xmlns="http://www.springframework.org/schema/beans"
xmlns:context="http://www.springframework.org/schema/context"
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-2.5.xsd
http://www.springframework.org/schema/context
http://www.springframework.org/schema/context/spring-context-3.0.xsd
http://www.springframework.org/schema/mvc
http://www.springframework.org/schema/mvc/spring-mvc-3.0.xsd">
<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>
<context:component-scan base-package="com.smiechmateusz.controller" />
<context:annotation-config />
<mvc:annotation-driven />
</beans>
HelloWorldController.java
package com.smiechmateusz.controller;
import javax.servlet.http.HttpServletRequest;
import javax.servlet.http.HttpServletResponse;
import org.springframework.stereotype.Controller;
import org.springframework.web.bind.annotation.RequestMapping;
import org.springframework.web.bind.annotation.RequestMethod;
import org.springframework.web.bind.annotation.ResponseBody;
import org.springframework.web.servlet.ModelAndView;
import org.springframework.web.servlet.mvc.AbstractController;
import com.smiechmateusz.dao.Foo;
@Controller
@RequestMapping("/")
public class HelloWorldController extends AbstractController{
@Override
protected ModelAndView handleRequestInternal(HttpServletRequest request,
HttpServletResponse response) throws Exception {
ModelAndView model = new ModelAndView("HelloWorldPage");
return model;
}
@RequestMapping(value="foobar.htm", method = RequestMethod.GET)
public @ResponseBody Foo getShopInJSON() {
Foo f = new Foo();
f.setX(1);
f.setY(2);
f.setDescription("desc");
return f;
}
}
Foo.java
package com.smiechmateusz.dao;
import java.io.Serializable;
import javax.persistence.Column;
import javax.persistence.Entity;
import javax.persistence.GeneratedValue;
import javax.persistence.Id;
import javax.persistence.Table;
@Entity
@Table(name="foobaz")
public class Foo implements Serializable
{
private int x, y;
String description;
int id;
@Column(name = "x")
public int getX() {
return x;
}
public void setX(int x) {
this.x = x;
}
@Column(name = "y")
public int getY() {
return y;
}
public void setY(int y) {
this.y = y;
}
@Column(name = "description")
public String getDescription() {
return description;
}
public void setDescription(String description) {
this.description = description;
}
@Id @GeneratedValue
@Column(name = "id")
public int getId() {
return id;
}
public void setId(int id) {
this.id = id;
}
}
Я уже пробовал добавить
<bean id="jsonConverter" class="org.springframework.http.converter.json.MappingJacksonHttpMessageConverter"></bean>
<bean class="org.springframework.web.servlet.mvc.annotation.AnnotationMethodHandlerAdapter">
<property name="messageConverters">
<list>
<ref bean="jsonConverter"/>
</list>
</property>
</bean>
для моего dispatcher-servlet.xml или изменения jakcson-all для jackson-asl и jackson-core-asl, но вывод был одинаков.
Ответы
Ответ 1
Accept: Текст /HTML, приложение/XHTML + XML, приложение /XML; д = 0,9,/; д = 0,8
Это должна быть проблема. JSON используется как application/json
. Если вы соответствующим образом установите заголовок Accept, вы должны получить правильный ответ. (Есть плагины браузера, которые позволяют устанавливать заголовки, мне нравится "Плакат" для Firefox лучше всего)
Ответ 2
Если вы используете Maven и последний код Джексона, вы можете удалить всю конфигурацию Jackson из ваших XML файлов конфигурации spring (вы будете по-прежнему нужен ведомый аннотациями тэг < mvc: annotation-driven/ > ) и просто добавьте некоторые зависимости Jackson к вашему файлу pom.xml. Ниже приведены примеры зависимостей. Это сработало для меня, и я использую:
- Apache Maven 3.0.4 (r1232337; 2012-01-17 01: 44: 56-0700)
- org.springframework version 3.1.2.RELEASE
-
spring -security version 3.1.0.RELEASE.
...<dependencies>
...
<dependency>
<groupId>com.fasterxml.jackson.core</groupId>
<artifactId>jackson-core</artifactId>
<version>2.2.3</version>
</dependency>
<dependency>
<groupId>com.fasterxml.jackson.core</groupId>
<artifactId>jackson-databind</artifactId>
<version>2.2.3</version>
</dependency>
<dependency>
<groupId>com.fasterxml.jackson.core</groupId>
<artifactId>jackson-annotations</artifactId>
<version>2.2.3</version>
</dependency>
...
</dependencies>...
Ответ 3
Еще один способ получить эту ошибку - создать класс без публичных членов. 406 неприемлемо - это довольно бесполезное сообщение об ошибке в этом сценарии.
Ответ 4
С помощью Spring 4 вы добавляете @EnableWebMvc
, например:
@Controller
@EnableWebMvc
@RequestMapping(value = "/articles/action", headers="Accept=*/*", produces="application/json")
public class ArticlesController {
}
Ответ 5
Ни один из ответов не помог мне.
Я читал десятки ответов Stackoverflow о 406 Not Acceptable, HttpMediaTypeNotAcceptableException, multipart file, ResponseBody, настройка Принимать заголовки, производит, потребляет и т.д.
У нас был Spring 4.2.4 с SpringBoot и Jackson, сконфигурированный в build.gradle:
compile "com.fasterxml.jackson.core:jackson-core:2.6.7"
compile "com.fasterxml.jackson.core:jackson-databind:2.6.7"
Все маршруты работали нормально в наших других контроллерах, и мы могли использовать GET, POST, PUT и DELETE. Затем я начал добавлять возможности многопользовательской загрузки файлов и создал новый контроллер. Маршруты GET работают нормально, но наши POST и DELETE этого не сделали. Независимо от того, как я пробовал разные решения отсюда в SO, я просто продолжал получать 406 недопустимых.
Тогда, наконец, я наткнулся на этот ответ:
Spring throw HttpMediaTypeNotAcceptableException: не удалось найти приемлемое представление из-за точки в URL-адресе
Прочитайте ответ Раниса и все комментарии.
Все это сводилось к нашим значениям @RequestMapping:
@RequestMapping(value = "/audio/{fileName:.+}", method = RequestMethod.POST, consumes="multipart/*")
public AudioFileDto insertAudio(@PathVariable String fileName, @RequestParam("audiofile") MultipartFile audiofile) {
return audioService.insert(fileName, audiofile);
}
@RequestMapping(value = "/audio/{fileName:.+}", method = RequestMethod.DELETE)
public Boolean deleteAudio(@PathVariable String fileName) {
return audioService.remove(fileName);
}
Часть {fileName:.+}
в значении @RequestMapping вызвала 406 недопустимых в нашем случае.
Вот код, который я добавил из ответа Раниса:
@Configuration
public class ContentNegotiationConfig extends WebMvcConfigurerAdapter {
@Override
void configureContentNegotiation(final ContentNegotiationConfigurer configurer) {
// Turn off suffix-based content negotiation
configurer.favorPathExtension(false);
}
}
EDIT 29 августа 2016 года:
У нас возникли проблемы с использованием configurer.favorPathExtension(false)
: статические изображения SVG перестали загружаться. После анализа мы обнаружили, что Spring начал отправлять SVG файлы обратно в пользовательский интерфейс с типом "application/octet-stream" типа контента вместо "image/svg + xml". Мы решили это, отправив fileName в качестве параметра запроса, например:
@RequestMapping(value = "/audio", method = RequestMethod.DELETE)
public Boolean deleteAudio(@RequestParam String fileName) {
return audioService.remove(fileName);
}
Мы также удалили configurer.favorPathExtension(false)
. Другой способ может заключаться в кодировании fileName в пути, но мы выбрали метод параметров запроса, чтобы избежать дополнительных побочных эффектов.
Ответ 6
Используйте зависимость ниже в вашей pom
<dependency>
<groupId>com.fasterxml.jackson.core</groupId>
<artifactId>jackson-databind</artifactId>
<version>2.5.3</version>
</dependency>
Ответ 7
Вам необходимо зарегистрировать привязку аннотации для Jackson в вашем spring -mvc-config.xml, например:
<!-- activates annotation driven binding -->
<mvc:annotation-driven ignoreDefaultModelOnRedirect="true" validator="validator">
<mvc:message-converters>
<bean class="org.springframework.http.converter.ResourceHttpMessageConverter"/>
<bean class="org.springframework.http.converter.xml.Jaxb2RootElementHttpMessageConverter"/>
<bean class="org.springframework.http.converter.json.MappingJacksonHttpMessageConverter"/>
</mvc:message-converters>
</mvc:annotation-driven>
Затем в вашем контроллере вы можете использовать:
@RequestMapping(value = "/your_url", method = RequestMethod.GET, produces = "application/json")
@ResponseBody
Ответ 8
Я полагаю, что проблема заключалась в использовании расширения *.htm в RequestMapping (foobar.htm). Попытайтесь изменить его на footer.json или что-то еще.
Ссылка на правильный ответ: fooobar.com/questions/117215/...
P.S.
В порядке Spring что-то делать по умолчанию, относительно того, что разработчики знают весь API Spring от A до Z. И тогда просто "406 неприемлемо" без каких-либо подробностей, а журналы Tomcat пусты
Ответ 9
См. проблему с расширением. В зависимости от расширения spring может определить тип содержимого. Если ваш URL-адрес заканчивается .com, он отправляет текст /html в качестве заголовка содержимого. Если вы хотите изменить это поведение Spring, используйте приведенный ниже код:
@Configuration
@Import(HibernateConfig.class)
@EnableWebMvc
// @EnableAsync()
// @EnableAspectJAutoProxy
@ComponentScan(basePackages = "com.azim.web.service.*", basePackageClasses = { WebSecurityConfig.class }, excludeFilters = { @ComponentScan.Filter(Configuration.class) })
public class WebConfig extends WebMvcConfigurerAdapter {
@Override
public void configureContentNegotiation(ContentNegotiationConfigurer configurer) {
configurer.favorPathExtension(false).favorParameter(true).parameterName("mediaType").ignoreAcceptHeader(true).useJaf(false)
.defaultContentType(MediaType.APPLICATION_JSON).mediaType("xml", MediaType.APPLICATION_XML).mediaType("json", MediaType.APPLICATION_JSON);
}
@Bean(name = "validator")
public Validator validator() {
return new LocalValidatorFactoryBean();
}
}
Здесь мы устанавливаем для параметра favorPathExtension значение false и Default Content-type для Application/json.
Примечание. Класс HibernateConfig содержит все beans.
Ответ 10
Попробуйте добавить
@RequestMapping(method = RequestMethod.GET,headers = {"Accept=text/xml, application/json"})
on getShopInJSON()
.
Это сработало для меня.
Ответ 11
Возможно, для всех полей вашего POJO нужны Getter и Setter.
Я исправил это в соответствии с этой проблемой.
ссылка: Spring MVC - HttpMediaTypeNotAcceptableException
И 406 не является полезным сообщением для исправления ошибки. Вы должны отлаживать коды и посмотреть, что такое Исключение на земле.
Ответ 12
это из-за того, что объект неприемлем для jsp... используйте его
добавить эту зависимость или любую другую переданную преобразованную строку json в jsp...
например
добавьте это в pom
<dependency>
<groupId>com.google.code.gson</groupId>
<artifactId>gson</artifactId>
<version>2.6.2</version>
</dependency>
и используйте такой код:
@RequestMapping(value="foobar.htm", method = RequestMethod.GET)
public @ResponseBody String getShopInJSON() {
Foo f = new Foo();
f.setX(1);
f.setY(2);
f.setDescription("desc");
return new Gson().toJson(f); //converted object into json string
}//return converted json string
Ответ 13
Сегодня я тоже пережил ту же проблему. В моем случае в web.xml
у меня есть
<servlet-mapping>
<servlet-name>spring</servlet-name>
<url-pattern>*.html</url-pattern>
</servlet-mapping>
мой url имеет расширение .html
. Пример: .../getUsers.html
. Но я возвращаю данные JSON в контроллер..html будет по умолчанию устанавливать тип accept как html.
Итак, я изменил на следующее:
web.xml:
<servlet-mapping>
<servlet-name>spring</servlet-name>
<url-pattern>*.html</url-pattern>
<url-pattern>*.json</url-pattern>
</servlet-mapping>
URL:
.../getUsers.json
Теперь все работает нормально. Надеюсь, что это поможет.
Ответ 14
Похоже, что вы пытаетесь создать/получить json-выход.
Я вижу две проблемы с вашим подходом.
1) Вы не указали приложение /json в заголовке Accept
2) Вам нужно указать output = "application/json" в @RequestMapping
Ответ 15
Есть еще один случай, когда этот статус будет возвращен: если планшет Jackson не может понять, как сериализовать ваш bean. Например, если у вас есть два метода доступа для одного и того же логического свойства, isFoo()
и getFoo()
.
удален getFoo()
и поместите isFoo()
. это сработало для меня.
Ответ 16
Я не мог видеть это как ответ здесь, поэтому я подумал, что хочу упомянуть, что я получил эту ошибку, используя spring 4.2, когда я случайно удалил getter/setter для класса, который я ожидал вернуть как Json.
Ответ 17
Значение My RequestMapping
заканчивалось .html, которое должно быть чем-то другим.
Я попытался изменить его на .json, и это сработало для меня.
Ответ 18
У меня есть аналогичная проблема и разрешена с помощью кода ниже
public class ProductList {
private List<Product> productList = new ArrayList<Product>();
@JsonProperty("data")
@JsonTypeInfo(use=JsonTypeInfo.Id.NAME, include=JsonTypeInfo.As.WRAPPER_OBJECT)
public List<Product> getProductList() {
return productList;
}
public void setProductList(List<Product> productList) {
this.productList = productList;
}
I am setting ProductList object in ResponseEntity object and returning from controller.
Ответ 19
Мой класс был аннотирован с помощью JsonSerialize, а параметр include был установлен в JsonSerialize.Inclusion.NON_DEFAULT
. Это заставило Джексона определить значения по умолчанию для каждого свойства bean. У меня было свойство bean, которое возвращало int. В моем случае проблема заключалась в том, что гейтер bean сделал вызов метода, который имеет тип возвращаемого возврата (т.е. Общий метод). По какой-то нечетной причине этот код скомпилирован; он не должен компилироваться, потому что вы не можете использовать int для типа возвращаемого возврата. Я изменил "int" на "Integer" для этого свойства bean, и у меня больше нет 406. Честно говоря, теперь код не получается скомпилировать, если я сменил Integer на int.