@ExceptionHandler не работает с @EnableWebMvc, HttpMediaTypeNotAcceptableException (406) возвращается для ResponseEntity - PullRequest
0 голосов
/ 24 августа 2018

У меня есть несколько веб-служб REST в Spring Boot (версия 1.5.13)

Я хотел бы использовать @ControllerAdvice для обработки исключений, выдаваемых контроллерами.

Для API testAниже класс @ControllerAdvice может захватывать и обрабатывать исключение, если id = 123, однако, если id не равен 123, моя программа не может преобразовать ResponseEntity в JSON и "org.springframework.web.HttpMediaTypeNotAcceptableException:Не удалось найти приемлемое представление (406) ".

Если я раскомментирую ниже @EnableWebMvc, ResponseEntity сможет правильно вернуть результат JSON, однако @ControllerAdvice не сможет перехватить исключение.

@RestController("SpocController")
//@EnableWebMvc
@RequestMapping(value = { "/testing" })
public class TestController 
{   
    @GetMapping(value = "/testA", produces = MediaType.APPLICATION_JSON_VALUE)
    public @ResponseBody ResponseEntity<Object> testA(@RequestParam String id) throws IOException   
    {       
        if(id.equals("123"))
        {
            throw new IOException("customized exception");
        }

        Map<String,Object> body = new HashMap<String, Object>();
        body.put("hello", "hihi");

        return new ResponseEntity<Object>(body, HttpStatus.OK);     
    }

    @GetMapping(value = "/testB", produces = MediaType.APPLICATION_JSON_VALUE)
    public @ResponseBody String testB(@RequestParam String id) throws IOException 
    {
        if(id.equals("123"))
        {
            throw new IOException("customized exception");
        }

        return "{ \"ok\":\"test ok\" }";
    }
}

Для API testB, поскольку тип возвращаемого значения - String, он мог возвращать строку в JSON, а @ControllerAdvice также может захватывать и обрабатывать исключение.

Это мой ControllerAdvice:

@ControllerAdvice
public class TestExceptionHandler
{
    @ResponseBody
    @ExceptionHandler(IOException.class)
    protected ResponseEntity<Object> handleIOException(IOException ex) 
    {   
        Map<String, Object> map = new HashMap<String, Object>();
        map.put("message", ex.getMessage());        

        return new ResponseEntity(map, HttpStatus.INTERNAL_SERVER_ERROR);       
    }   
}

Это HttpMediaTypeNotAcceptableException (406) I menвыше.

org.springframework.web.HttpMediaTypeNotAcceptableException: Could not find acceptable representation
    at org.springframework.web.servlet.mvc.method.annotation.AbstractMessageConverterMethodProcessor.writeWithMessageConverters(AbstractMessageConverterMethodProcessor.java:259)
    at org.springframework.web.servlet.mvc.method.annotation.HttpEntityMethodProcessor.handleReturnValue(HttpEntityMethodProcessor.java:208)
    at org.springframework.web.method.support.HandlerMethodReturnValueHandlerComposite.handleReturnValue(HandlerMethodReturnValueHandlerComposite.java:81)
    at org.springframework.web.servlet.mvc.method.annotation.ServletInvocableHandlerMethod.invokeAndHandle(ServletInvocableHandlerMethod.java:113)
    at org.springframework.web.servlet.mvc.method.annotation.RequestMappingHandlerAdapter.invokeHandlerMethod(RequestMappingHandlerAdapter.java:827)
    at org.springframework.web.servlet.mvc.method.annotation.RequestMappingHandlerAdapter.handleInternal(RequestMappingHandlerAdapter.java:738)
    at org.springframework.web.servlet.mvc.method.AbstractHandlerMethodAdapter.handle(AbstractHandlerMethodAdapter.java:85)
    at org.springframework.web.servlet.DispatcherServlet.doDispatch(DispatcherServlet.java:967)
    at org.springframework.web.servlet.DispatcherServlet.doService(DispatcherServlet.java:901)
    at org.springframework.web.servlet.FrameworkServlet.processRequest(FrameworkServlet.java:970)
    at org.springframework.web.servlet.FrameworkServlet.doGet(FrameworkServlet.java:861)
    at javax.servlet.http.HttpServlet.service(HttpServlet.java:635)
    at org.springframework.web.servlet.FrameworkServlet.service(FrameworkServlet.java:846)
    at javax.servlet.http.HttpServlet.service(HttpServlet.java:742)
    at org.apache.catalina.core.ApplicationFilterChain.internalDoFilter(ApplicationFilterChain.java:231)
    at org.apache.catalina.core.ApplicationFilterChain.doFilter(ApplicationFilterChain.java:166)
    at org.apache.tomcat.websocket.server.WsFilter.doFilter(WsFilter.java:52)
    at org.apache.catalina.core.ApplicationFilterChain.internalDoFilter(ApplicationFilterChain.java:193)
    at org.apache.catalina.core.ApplicationFilterChain.doFilter(ApplicationFilterChain.java:166)
    at org.springframework.web.filter.RequestContextFilter.doFilterInternal(RequestContextFilter.java:99)
    at org.springframework.web.filter.OncePerRequestFilter.doFilter(OncePerRequestFilter.java:107)
    at org.apache.catalina.core.ApplicationFilterChain.internalDoFilter(ApplicationFilterChain.java:193)
    at org.apache.catalina.core.ApplicationFilterChain.doFilter(ApplicationFilterChain.java:166)
    at org.springframework.web.filter.HttpPutFormContentFilter.doFilterInternal(HttpPutFormContentFilter.java:109)
    at org.springframework.web.filter.OncePerRequestFilter.doFilter(OncePerRequestFilter.java:107)
    at org.apache.catalina.core.ApplicationFilterChain.internalDoFilter(ApplicationFilterChain.java:193)
    at org.apache.catalina.core.ApplicationFilterChain.doFilter(ApplicationFilterChain.java:166)
    at org.springframework.web.filter.HiddenHttpMethodFilter.doFilterInternal(HiddenHttpMethodFilter.java:81)
    at org.springframework.web.filter.OncePerRequestFilter.doFilter(OncePerRequestFilter.java:107)
    at org.apache.catalina.core.ApplicationFilterChain.internalDoFilter(ApplicationFilterChain.java:193)
    at org.apache.catalina.core.ApplicationFilterChain.doFilter(ApplicationFilterChain.java:166)
    at org.springframework.web.filter.CharacterEncodingFilter.doFilterInternal(CharacterEncodingFilter.java:197)
    at org.springframework.web.filter.OncePerRequestFilter.doFilter(OncePerRequestFilter.java:107)
    at org.apache.catalina.core.ApplicationFilterChain.internalDoFilter(ApplicationFilterChain.java:193)
    at org.apache.catalina.core.ApplicationFilterChain.doFilter(ApplicationFilterChain.java:166)
    at org.apache.catalina.core.StandardWrapperValve.invoke(StandardWrapperValve.java:198)
    at org.apache.catalina.core.StandardContextValve.invoke(StandardContextValve.java:96)
    at org.apache.catalina.authenticator.AuthenticatorBase.invoke(AuthenticatorBase.java:496)
    at org.apache.catalina.core.StandardHostValve.invoke(StandardHostValve.java:140)
    at org.apache.catalina.valves.ErrorReportValve.invoke(ErrorReportValve.java:81)
    at org.apache.catalina.core.StandardEngineValve.invoke(StandardEngineValve.java:87)
    at org.apache.catalina.connector.CoyoteAdapter.service(CoyoteAdapter.java:342)
    at org.apache.coyote.http11.Http11Processor.service(Http11Processor.java:803)
    at org.apache.coyote.AbstractProcessorLight.process(AbstractProcessorLight.java:66)
    at org.apache.coyote.AbstractProtocol$ConnectionHandler.process(AbstractProtocol.java:790)
    at org.apache.tomcat.util.net.NioEndpoint$SocketProcessor.doRun(NioEndpoint.java:1468)
    at org.apache.tomcat.util.net.SocketProcessorBase.run(SocketProcessorBase.java:49)
    at java.util.concurrent.ThreadPoolExecutor.runWorker(ThreadPoolExecutor.java:1149)
    at java.util.concurrent.ThreadPoolExecutor$Worker.run(ThreadPoolExecutor.java:624)
    at org.apache.tomcat.util.threads.TaskThread$WrappingRunnable.run(TaskThread.java:61)
    at java.lang.Thread.run(Thread.java:748)

1) Независимо от того, куда я поместил @EnableWebMvc (в @RestController, @ControllerAdvice или другие классы), мой @ExceptionHandler не сможет перехватить ни одно исключение.Любая идея, почему это происходит?

2) Я бы хотел, чтобы мой API testA работал, так как я действительно хочу вернуть тип ResponseEntity, и вызванное им исключение может быть перехвачено @ControllerAdvice.Любые советы приветствуются.

Заранее спасибо.

Ответы [ 3 ]

0 голосов
/ 26 августа 2018
  1.                     @RestController
                        @RequestMapping(value = { "/testing" })
                        public class ExceptionController {
                             @GetMapping(value = "/testA", produces = MediaType.APPLICATION_JSON_VALUE)
                                public  ResponseEntity<Object> testA(@RequestParam String id) throws IOException   
                                {       
                                    if(id.equals("123"))
                                    {
                                        throw new IOException("customized exception");
                                    }
    
                                    Map<String,Object> body = new HashMap<String, Object>();
                                    body.put("hello", "hihi");
    
                                    return new ResponseEntity<Object>(body, HttpStatus.OK);     
                                }
                             @GetMapping(value = "/testB", produces = MediaType.APPLICATION_JSON_VALUE)
                                public  String testB(@RequestParam String id) throws IOException 
                                {
                                    if(id.equals("123"))
                                    {
                                        throw new IOException("customized exception");
                                    }
    
                                    return "{ \"ok\":\"test ok\" }";
                                }
    
                        }
    
    

2. измените свой совет контроллера -:

                    package com.ad.blog.controller;

                    import java.io.IOException;
                    import java.util.Optional;

                    import org.springframework.hateoas.VndErrors;
                    import org.springframework.http.HttpStatus;
                    import org.springframework.http.ResponseEntity;
                    import org.springframework.web.bind.annotation.ExceptionHandler;
                    import org.springframework.web.bind.annotation.RequestMapping;

                    @org.springframework.web.bind.annotation.ControllerAdvice
                    @RequestMapping(produces = "application/vnd.error+json")
                    public class ControllerAdvice {
                        @ExceptionHandler(IOException.class) public ResponseEntity < VndErrors > notFoundException(final IOException e) {
                            return error(e, HttpStatus.NOT_FOUND, e.getMessage());
                        }
                        private ResponseEntity < VndErrors > error(final Exception exception, final HttpStatus httpStatus, final String logRef) {
                            final String message = Optional.of(exception.getMessage()).orElse(exception.getClass().getSimpleName());
                            return new ResponseEntity < > (new VndErrors(logRef, message), httpStatus);
                        }
                    }
  1. вам нужно добавить зависимость hateos в ваш файл pom.xml.
                    <dependency>
                                <groupId>org.springframework.boot</groupId>
                                <artifactId>spring-boot-starter-hateoas</artifactId>
                            </dependency>
  1. @ Аннотация EnableWebMVc используется для включения конфигурации MVC Java, которая переопределяет конфигурацию по умолчанию DispatcherServlet. Если вы хотите переопределить конфигурацию по умолчанию, а затем создать класс -:
        @Configuration
        @EnableWebMvc
        public class WebConfig extends WebMvcConfigurerAdapter {


        }
0 голосов
/ 03 сентября 2018

Нашли виновника, прокомментировав следующий раздел кода, все работает как шарм.В любом случае, спасибо всем!

public HandlerExceptionResolver handlerExceptionResolver()
{
    return new HandlerExceptionResolver()
    {
        @Override
        public ModelAndView resolveException(HttpServletRequest request, HttpServletResponse response, Object handler, Exception exception)
        {
            return null;
        }
    };
}
0 голосов
/ 25 августа 2018

С @RestController,

Нет необходимости использовать @ResponseBody, вы можете просмотреть документ для более подробной информации.

-Убедитесь в том, что ваше отображение запросов работает и что-то регистрируйте.

-Попробуйте @RestControllerAdvice для @ RestController

Добро пожаловать на сайт PullRequest, где вы можете задавать вопросы и получать ответы от других членов сообщества.
...