Spring MVC - HttpMediaTypeNotAcceptableException
Я продолжаю получать эту ошибку HttpMediaTypeNotAcceptableException для запросов AJAX при использовании с Spring MVC и JSON. Полная трассировка стека ошибки...
org.springframework.web.HttpMediaTypeNotAcceptableException: Could not find acceptable representation
org.springframework.web.servlet.mvc.annotation.AnnotationMethodHandlerAdapter$ServletHandlerMethodInvoker.writeWithMessageConverters(AnnotationMethodHandlerAdapter.java:1032)
org.springframework.web.servlet.mvc.annotation.AnnotationMethodHandlerAdapter$ServletHandlerMethodInvoker.handleResponseBody(AnnotationMethodHandlerAdapter.java:972)
org.springframework.web.servlet.mvc.annotation.AnnotationMethodHandlerAdapter$ServletHandlerMethodInvoker.getModelAndView(AnnotationMethodHandlerAdapter.java:921)
org.springframework.web.servlet.mvc.annotation.AnnotationMethodHandlerAdapter.invokeHandlerMethod(AnnotationMethodHandlerAdapter.java:438)
org.springframework.web.servlet.mvc.annotation.AnnotationMethodHandlerAdapter.handle(AnnotationMethodHandlerAdapter.java:424)
org.springframework.web.servlet.DispatcherServlet.doDispatch(DispatcherServlet.java:863)
org.springframework.web.servlet.DispatcherServlet.doService(DispatcherServlet.java:792)
org.springframework.web.servlet.FrameworkServlet.processRequest(FrameworkServlet.java:851)
org.springframework.web.servlet.FrameworkServlet.doGet(FrameworkServlet.java:756)
javax.servlet.http.HttpServlet.service(HttpServlet.java:617)
javax.servlet.http.HttpServlet.service(HttpServlet.java:717)
Небольшое приближение, которое я сделал, показывает, что запрос должен содержать что-то вроде "accept: application/json", которое действительно имеет… вот заголовки запроса от firebug..
Response Headers
Server Apache-Coyote/1.1
Content-Type text/html;charset=utf-8
Content-Length 2503
Date Thu, 25 Aug 2011 21:00:05 GMT
Connection close
Request Headers
Host localhost:8080
User-Agent Mozilla/5.0 (Windows; U; Windows NT 5.1; en-US; rv:1.9.2.20) Gecko/20110803 Firefox/3.6.20 (.NET CLR 3.5.30729)
Accept application/json, text/javascript, */*; q=0.01
Accept-Language en-us,en;q=0.5
Accept-Encoding gzip,deflate
Accept-Charset ISO-8859-1,utf-8;q=0.7,*;q=0.7
Keep-Alive 115
Connection keep-alive
X-Requested-With XMLHttpRequest
Referer http://localhost:8080/legaldirectory/index.html
Cookie JSESSIONID=5C97DA19AED4D5FA17F4A58470FAA93B
Теперь я полностью потерян в том, что здесь происходит.. что еще может пойти не так, чтобы получить эту ошибку...
13 ответов
Убедитесь, что в вашем Spring XML-файле есть следующее:
<context:annotation-config/>
<bean id="jacksonMessageConverter" class="org.springframework.http.converter.json.MappingJacksonHttpMessageConverter"></bean>
<bean class="org.springframework.web.servlet.mvc.annotation.AnnotationMethodHandlerAdapter">
<property name="messageConverters">
<list>
<ref bean="jacksonMessageConverter"/>
</list>
</property>
</bean>
и все предметы вашего POJO должны иметь геттеры / сеттеры. Надеюсь, поможет
От: http://georgovassilis.blogspot.ca/2015/10/spring-mvc-rest-controller-says-406.html
Вы получили этот Spring @RestController и сопоставили URL-адрес, содержащий электронное письмо, как часть пути URL-адреса. Вы ловко обошли проблему с усечением точек [1] и готовы к работе. И вдруг на некоторых URL Spring возвращает 406 [2], в котором говорится, что браузер запросил определенный тип контента, и Spring не может сериализовать ответ на этот тип контента. Дело в том, что вы годами работали с приложениями Spring, правильно выполнили все декларации MVC, включили Джексона и, по сути, застряли. Что еще хуже, эта ошибка будет появляться только в некоторых электронных письмах в URL-пути, особенно в тех, которые заканчиваются доменом ".com".
@RequestMapping(value = "/agenda/{email:.+}", method = RequestMethod.GET)
public List<AgendaEntryDTO> checkAgenda(@PathVariable("email") String email)
Проблема [3] довольно сложна: сервер приложений выполняет некоторое согласование контента и убеждает Spring в том, что браузер запросил контент "application/x-msdownload", несмотря на то, что это нигде не происходит в запросе, который браузер фактически отправил.
Решение состоит в том, чтобы указать менеджер согласования контента для контекста веб-приложения:
<mvc:annotation-driven enable-matrix-variables="true"
content-negotiation-manager="contentNegotiationManager" />
<bean id="contentNegotiationManager"
class="org.springframework.web.accept.ContentNegotiationManagerFactoryBean">
<property name="defaultContentType" value="application/json" />
<property name="favorPathExtension" value="false" />
<property name="favorParameter" value="false" />
<property name="parameterName" value="mediaType" />
<property name="ignoreAcceptHeader" value="false" />
<property name="useJaf" value="false" />
</bean>
Убедитесь, что вы добавили обе банки Джексона в classpath:
- ДЖЕКСОН-ядро-ASL-x.jar
- ДЖЕКСОН-картограф-ASL-x.jar
Кроме того, в вашем Spring XML-файле должно быть следующее:
<mvc:annotation-driven />
Поскольку это первое попадание Google для "HttpMediaTypeNotAcceptableException", я хотел бы добавить еще одну проблему, на которую я наткнулся, что также привело к HttpMediaTypeNotAcceptableException.
В моем случае это был контроллер, который "производит", например:
@RequestMapping(path = "/mypath/{filename}", method = RequestMethod.GET,
produces = { MediaType.APPLICATION_XML_VALUE }
потому что я хотел обслужить файл XML. В то же время я использую класс с @ControllerAdvice для перехвата исключений, например, если запрошенный файл не найден. Обработчик исключений возвращал JSON, чтобы клиентское (угловое) приложение могло отображать сообщение об ошибке где-то в SPA.
Теперь контроллер хотел вернуть XML, но обработчик исключений возвращал JSON, поэтому было вызвано исключение HttpMediaTypeNotAcceptableException. Я решил это, добавив JSON как можно более "производящее" значение:
produces = { MediaType.APPLICATION_XML_VALUE, MediaType.APPLICATION_JSON_VALUE}
Надеюсь, это поможет кому-то еще.:-)
Просто важно помнить: версии Spring до 3.1.2 совместимы с JACKSON 1.x, а НЕ с JACKSON 2.x. Это происходит потому, что при переходе с JACKSON 1.x на 2.x имена пакетов классов были изменены. В JACKSON 1.x классы находятся под org.codehaus.jackson, а в JACKSON 2.x они находятся под com.fasterxml.jackson.
Чтобы решить эту проблему, начиная с Spring 3.1.2, они добавили новый MappingJackson2HttpMessageConverter вместо MappingJacksonHttpMessageConverter.
Вы можете найти более подробную информацию о проблемах совместимости по этой ссылке: аннотации Джексона игнорируются в Spring
В моем случае favPathExtension(false) помог мне
@Configuration
public class WebMvcConfiguration extends WebMvcConfigurerAdapter {
@Override
public void configurePathMatch(PathMatchConfigurer configurer) {
configurer
.setUseSuffixPatternMatch(false); // to use special character in path variables, for example, `email@dom.com`
}
@Override
public void configureContentNegotiation(ContentNegotiationConfigurer configurer) {
configurer
.favorPathExtension(false); // to avoid HttpMediaTypeNotAcceptableException on standalone tomcat
}
}
В вашем классе Spring @Configuration, который расширяет WebMvcConfigurerAdapter, переопределите метод configureMessageConverters, например:
@Override
public void configureMessageConverters(List<HttpMessageConverter<?>> converters) {
// http
HttpMessageConverter converter = new StringHttpMessageConverter();
converters.add(converter);
logger.info("HttpMessageConverter added");
// string
converter = new FormHttpMessageConverter();
converters.add(converter);
logger.info("FormHttpMessageConverter added");
// json
converter = new MappingJackson2HttpMessageConverter();
converters.add(converter);
logger.info("MappingJackson2HttpMessageConverter added");
}
И ошибка исчезнет.
У меня была та же проблема, но я понял, что в основном то, что происходит, когда Spring пытается отобразить ответ, попытается сериализовать его в соответствии с типом носителя, который вы указали, и используя методы getter и setter в вашем классе.
до того, как мой класс выглядел как ниже
public class MyRestResponse{
private String message;
}
решение выглядит как ниже
public class MyRestResponse{
private String message;
public void setMessage(String msg){
this.message = msg;
}
public String getMessage(){
return this.message;
}
}
вот как это работает для меня
Попробуйте добавить файлы jaxson-databind в pom.xml
`<dependency>
<groupId>com.fasterxml.jackson.core</groupId>
<artifactId>jackson-databind</artifactId>
<version>${jackson.databind-version}</version>
</dependency>`
А также produces ={MediaType.APPLICATION_JSON_VALUE, MediaType.APPLICATION_XML_VALUE}
в @RequestMapping
Es. @RequestMapping(value = "/api/xxx/{akey}/{md5}", produces ={MediaType.APPLICATION_JSON_VALUE, MediaType.APPLICATION_XML_VALUE}
@RequestMapping(value = "/api/contrassegno/{akey}/{md5}", produces ={MediaType.APPLICATION_JSON_VALUE, MediaType.APPLICATION_XML_VALUE},...
Для меня проблемой был URL, к которому я пытался получить доступ. У меня был URL, как это:
{{ip}}:{{port}}/{{path}}/person.html
Когда вы заканчиваете URL с помощью.html, это означает, что вы не примете никакой другой полезной нагрузки, кроме html. Мне нужно было удалить.html с конца, чтобы моя конечная точка работала правильно. (Вы также можете добавить.json в конце URL, и это тоже будет работать)
Кроме того, ваш URL-шаблон в web.xml
должны быть правильно настроены, чтобы вы могли получить доступ к ресурсу. Для меня это
<servlet-mapping>
<servlet-name>spring-mvc</servlet-name>
<url-pattern>/</url-pattern>
</servlet-mapping>
Ранее у меня было *.html*
и это мешало мне получить доступ к конечной точке.
Как подсказал Алекс в одном из ответов, вы можете использовать ContentNegotiationManagerFactoryBean, чтобы установить тип контента по умолчанию равным "application/json", но я чувствовал, что такой подход не для меня.
Я пытался опубликовать форму в таком методе
@RequestMapping(value = "/post/to/me", method = RequestMethod.POST, produces = MediaType.APPLICATION_JSON_VALUE, consumes = MediaType.MULTIPART_FORM_DATA_VALUE)
public @ResponseBody MyJsonPOJO handlePostForm(@Valid @ModelAttribute("form") ValidateMeForm form, BindingResult bindingResult) throws ApiException {
Вместо этого я решил изменить заголовок "Accept" запроса из браузера на "application/json", тем самым заставив SpringMVC найти мой метод.
Использование (еще не завершенного) API Javascript Fetch:
var form = new FormData();
form.append("myData", "data");
let fetchConfig = {
method: "POST",
body: form,
headers: {"Accept": "application/json"}
};
fetch("/post/to/me", fetchConfig)
.then(... // Javascript Promise API here
И вуаля! Теперь SpringMVC находит метод, проверяет форму и позволяет вам возвращать JSON POJO.
Больше комментариев, чем ответа...
Я использую Lombok, и я разрабатывал (очень) скелетный API, и мой ответ DTO не имел никаких полей (пока), и я получил HttpMediaTypeNotAcceptableException
ошибка при запуске моих интеграционных тестов.
Добавление поля в ответ DTO устранило проблему.
В моем случае
{"timestamp":1537542856089,"status":406,"error":"Not Acceptable","exception":"org.springframework.web.HttpMediaTypeNotAcceptableException","message":"Could not find acceptable representation","path":"/a/101.xml"}
было вызвано:
path = "/path/{VariableName}"
но я передавал VariableName с суффиксом, например, "abc.xml", который заставляет его интерпретировать.xml как своего рода запрос формата. Смотрите ответы там.
В моем случае просто добавьте @ResponseBody
аннотация для решения этой проблемы.