Подтвердить что ты не робот

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, но вывод был одинаков.

4b9b3361

Ответ 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.