`RestControllerAdvice`在Spring Boot 2.3中未被调用。

huangapple 未分类评论46阅读模式
英文:

RestControllerAdvice not called with Spring Boot 2.3

问题

  1. 我想要将我的应用程序从Spring Boot 2.2.8更新到2.3.1
  2. 在更新后运行应用程序时,我的globalExceptionHandler将不再正常工作。
  3. 我的情况是在请求中存在无效的JSON主体时处理错误。
  4. 应用程序
  5. // 省略部分代码...
  6. 控制器
  7. // 省略部分代码...
  8. 异常处理器
  9. // 省略部分代码...
  10. 自更新后,我没有更改任何代码。响应代码仍然是之前的400 Bad Request,但响应主体为空,并且我的处理程序不会被调用。
  11. 日志:
  12. > 13:13:53.987 [http-nio-8080-exec-3] TRACE org.hibernate.internal.SessionImpl - 已打开会话 [6570b352-8067-49d2-bc46-c503917f4c8d],时间戳为:1595502833972
  13. 13:13:53.991 [http-nio-8080-exec-3] TRACE o.s.t.s.TransactionSynchronizationManager - 将值 [org.springframework.orm.jpa.EntityManagerHolder@72dd1712] 绑定到键 [org.springframework.orm.jpa.LocalContainerEntityManagerFactoryBean@38c55a8a],线程为 [http-nio-8080-exec-3]
  14. 13:13:54.011 [http-nio-8080-exec-3] DEBUG o.s.w.s.m.m.a.ServletInvocableHandlerMethod - 无法解析参数 [0]:public com.schaerer.coffeelink.remote.boundary.v1.dto.BidiCommandReturnDto com.schaerer.coffeelink.remote.boundary.v1.BidiCommandResource.applyBidiCommand(com.schaerer.coffeelink.remote.boundary.v1.dto.BidiCommandDto) 中的参数 [0]:JSON解析错误:意外的字符('\'(代码92)):期望双引号以开始字段名称;嵌套异常为 com.fasterxml.jackson.databind.JsonMappingException:意外的字符('\'(代码92)):期望双引号以开始字段名称
  15. at [Source: (PushbackInputStream); line: 8, column: 54](通过引用链:com.schaerer.coffeelink.remote.boundary.v1.dto.BidiCommandDto["parameters"])
  16. 13:13:54.012 [http-nio-8080-exec-3] TRACE o.s.b.f.s.DefaultListableBeanFactory - 返回单例bean 'globalExceptionHandler' 的缓存实例
  17. 13:13:54.012 [http-nio-8080-exec-3] DEBUG o.s.w.s.m.m.a.ExceptionHandlerExceptionResolver - 使用 @ExceptionHandler com.schaerer.coffeelink.remote.boundary.v1.GlobalExceptionHandler#handleException(Exception, WebRequest)
  18. 13:13:54.013 [http-nio-8080-exec-3] TRACE o.s.w.s.m.m.a.ServletInvocableHandlerMethod - 参数:[org.springframework.http.converter.HttpMessageNotReadableExceptionJSON解析错误:意外的字符('\'(代码92)):期望双引号以开始字段名称;嵌套异常为 com.fasterxml.jackson.databind.JsonMappingException:意外的字符('\'(代码92))):期望双引号以开始字段名称
  19. at [Source: (PushbackInputStream); line: 8, column: 54](通过引用链:com.schaerer.coffeelink.remote.boundary.v1.dto.BidiCommandDto["parameters"]),ServletWebRequesturi=/v1/remote/;client=0:0:0:0:0:0:0:1]
  20. 13:13:54.015 [http-nio-8080-exec-3] DEBUG o.s.w.s.m.m.a.HttpEntityMethodProcessor - 不匹配 [*/*],支持的:[]
  21. 13:13:54.016 [http-nio-8080-exec-3] DEBUG o.s.w.s.m.m.a.ExceptionHandlerExceptionResolver - 已解析 [org.springframework.http.converter.HttpMessageNotReadableException:JSON解析错误:意外的字符('\'(代码92)):期望双引号以开始字段名称;嵌套异常为 com.fasterxml.jackson.databind.JsonMappingException:意外的字符('\'(代码92))):期望双引号以开始字段名称
  22. at [Source: (PushbackInputStream); line: 8, column: 54](通过引用链:com.schaerer.coffeelink.remote.boundary.v1.dto.BidiCommandDto["parameters"])]
  23. 13:13:54.016 [http-nio-8080-exec-3] TRACE o.s.web.servlet.DispatcherServlet - 无视图渲染,返回空的 ModelAndView。
  24. 有什么想法吗?
英文:

I want to update my application from Spring Boot 2.2.8 to 2.3.1
When I run the application after the update, my globalExceptionHandler won't work anymore correctly.
My case was to handle the error, when a invalid JSON-Body is in the request.

Application

  1. package com.schaerer.coffeelink.remote;
  2. import lombok.extern.slf4j.Slf4j;
  3. import org.springframework.boot.SpringApplication;
  4. import org.springframework.boot.autoconfigure.SpringBootApplication;
  5. import org.springframework.kafka.annotation.EnableKafka;
  6. @Slf4j
  7. @SpringBootApplication
  8. @EnableKafka
  9. public class Application {
  10. public static void main(final String[] args) { //NOSONAR
  11. SpringApplication.run(Application.class, args);
  12. log.info("Started application. Swagger available at http://localhost:8080/swagger-ui.html");
  13. }
  14. }

Controller

  1. package com.schaerer.coffeelink.remote.boundary.v1;
  2. import com.fasterxml.jackson.databind.ObjectMapper;
  3. import com.schaerer.coffeelink.remote.boundary.v1.dto.*;
  4. import com.schaerer.coffeelink.remote.controller.v1.ActionController;
  5. import com.schaerer.coffeelink.remote.controller.v1.BidiCommandController;
  6. import io.swagger.v3.oas.annotations.Operation;
  7. import io.swagger.v3.oas.annotations.Parameter;
  8. import io.swagger.v3.oas.annotations.tags.Tag;
  9. import lombok.RequiredArgsConstructor;
  10. import lombok.extern.slf4j.Slf4j;
  11. import org.springframework.beans.factory.annotation.Autowired;
  12. import org.springframework.http.HttpStatus;
  13. import org.springframework.http.MediaType;
  14. import org.springframework.http.ResponseEntity;
  15. import org.springframework.validation.annotation.Validated;
  16. import org.springframework.web.bind.annotation.*;
  17. import org.springframework.web.multipart.MultipartFile;
  18. @Slf4j
  19. @RestController
  20. @Validated
  21. @RequestMapping(value = "v1/remote", produces = MediaType.APPLICATION_JSON_VALUE)
  22. @Tag(name = "remote-bidi-command")
  23. @RequiredArgsConstructor(onConstructor = @__(@Autowired))
  24. public class BidiCommandResource {
  25. private final BidiCommandController bidiCommandController;
  26. private final ActionController actionController;
  27. private final ObjectMapper mapper;
  28. @ExceptionHandler({IllegalArgumentException.class})
  29. public ResponseEntity<Object> handleException(final IllegalArgumentException illegalArgumentException) {
  30. return new ResponseEntity<>(illegalArgumentException.getMessage(), HttpStatus.BAD_REQUEST);
  31. }
  32. @PostMapping
  33. @Operation(summary = "apply bidi command 2.0")
  34. public BidiCommandReturnDto applyBidiCommand(@Parameter(name = "applyBidiCommand")
  35. @Validated({ActionDto.CreateValidation.class, BidiCommandDto.CreateValidation.class})
  36. @RequestBody final BidiCommandDto bidiCommandDto) {
  37. return bidiCommandController.applyBidiCommand(bidiCommandDto);
  38. }
  39. }

ExceptionHandler

  1. package com.schaerer.coffeelink.remote.boundary.v1;
  2. import com.fasterxml.jackson.databind.JsonMappingException;
  3. import lombok.extern.slf4j.Slf4j;
  4. import org.springframework.http.HttpHeaders;
  5. import org.springframework.http.HttpStatus;
  6. import org.springframework.http.ResponseEntity;
  7. import org.springframework.web.bind.annotation.ExceptionHandler;
  8. import org.springframework.web.bind.annotation.RestControllerAdvice;
  9. import org.springframework.web.context.request.WebRequest;
  10. import org.springframework.web.servlet.mvc.method.annotation.ResponseEntityExceptionHandler;
  11. @Slf4j
  12. @RestControllerAdvice
  13. public class GlobalExceptionHandler extends ResponseEntityExceptionHandler {
  14. @ExceptionHandler(value = {JsonMappingException.class})
  15. public ResponseEntity<Object> handleJacksonError(final JsonMappingException ex, final WebRequest request) {
  16. log.error("Cannot parse request. {}", request.getDescription(true), ex);
  17. return handleExceptionInternal(ex, ex.getMessage(), new HttpHeaders(), HttpStatus.BAD_REQUEST, null);
  18. }
  19. }

I haven't change any code since the update. The response code is as before 400 Bad Request but the response body is empty and my handler isn't called.

Logs:

> 13:13:53.987 [http-nio-8080-exec-3] TRACE org.hibernate.internal.SessionImpl - Opened Session [6570b352-8067-49d2-bc46-c503917f4c8d] at timestamp: 1595502833972
13:13:53.991 [http-nio-8080-exec-3] TRACE o.s.t.s.TransactionSynchronizationManager - Bound value [org.springframework.orm.jpa.EntityManagerHolder@72dd1712] for key [org.springframework.orm.jpa.LocalContainerEntityManagerFactoryBean@38c55a8a] to thread [http-nio-8080-exec-3]
13:13:54.011 [http-nio-8080-exec-3] DEBUG o.s.w.s.m.m.a.ServletInvocableHandlerMethod - Could not resolve parameter [0] in public com.schaerer.coffeelink.remote.boundary.v1.dto.BidiCommandReturnDto com.schaerer.coffeelink.remote.boundary.v1.BidiCommandResource.applyBidiCommand(com.schaerer.coffeelink.remote.boundary.v1.dto.BidiCommandDto): JSON parse error: Unexpected character ('' (code 92)): was expecting double-quote to start field name; nested exception is com.fasterxml.jackson.databind.JsonMappingException: Unexpected character ('' (code 92)): was expecting double-quote to start field name
at [Source: (PushbackInputStream); line: 8, column: 54] (through reference chain: com.schaerer.coffeelink.remote.boundary.v1.dto.BidiCommandDto["parameters"])
13:13:54.012 [http-nio-8080-exec-3] TRACE o.s.b.f.s.DefaultListableBeanFactory - Returning cached instance of singleton bean 'globalExceptionHandler'
13:13:54.012 [http-nio-8080-exec-3] DEBUG o.s.w.s.m.m.a.ExceptionHandlerExceptionResolver - Using @ExceptionHandler com.schaerer.coffeelink.remote.boundary.v1.GlobalExceptionHandler#handleException(Exception, WebRequest)
13:13:54.013 [http-nio-8080-exec-3] TRACE o.s.w.s.m.m.a.ServletInvocableHandlerMethod - Arguments: [org.springframework.http.converter.HttpMessageNotReadableException: JSON parse error: Unexpected character ('' (code 92)): was expecting double-quote to start field name; nested exception is com.fasterxml.jackson.databind.JsonMappingException: Unexpected character ('' (code 92)): was expecting double-quote to start field name
at [Source: (PushbackInputStream); line: 8, column: 54] (through reference chain: com.schaerer.coffeelink.remote.boundary.v1.dto.BidiCommandDto["parameters"]), ServletWebRequest: uri=/v1/remote/;client=0:0:0:0:0:0:0:1]
13:13:54.015 [http-nio-8080-exec-3] DEBUG o.s.w.s.m.m.a.HttpEntityMethodProcessor - No match for [/], supported: []
13:13:54.016 [http-nio-8080-exec-3] DEBUG o.s.w.s.m.m.a.ExceptionHandlerExceptionResolver - Resolved [org.springframework.http.converter.HttpMessageNotReadableException: JSON parse error: Unexpected character ('' (code 92)): was expecting double-quote to start field name; nested exception is com.fasterxml.jackson.databind.JsonMappingException: Unexpected character ('' (code 92)): was expecting double-quote to start field name
at [Source: (PushbackInputStream); line: 8, column: 54] (through reference chain: com.schaerer.coffeelink.remote.boundary.v1.dto.BidiCommandDto["parameters"])]
13:13:54.016 [http-nio-8080-exec-3] TRACE o.s.web.servlet.DispatcherServlet - No view rendering, null ModelAndView returned.

Any ideas?

答案1

得分: 2

我发现了一个解决方案,当我移除了ResponseEntityExceptionHandler扩展后。以下是我的解决方案:

  1. @Slf4j
  2. @RestControllerAdvice
  3. public class GlobalExceptionHandler {
  4. @ExceptionHandler(value = {HttpMessageNotReadableException.class})
  5. public ResponseEntity<Object> handleJacksonError(final JsonMappingException ex, final WebRequest request) {
  6. log.error("无法解析请求。 {}", request.getDescription(true), ex);
  7. return new ResponseEntity<>(ex.getMessage(), HttpStatus.BAD_REQUEST);
  8. }
  9. }
英文:

I've found a solution when I remove the ResponseEntityExceptionHandler extension.
Here my solution:

  1. @Slf4j
  2. @RestControllerAdvice
  3. public class GlobalExceptionHandler {
  4. @ExceptionHandler(value = {HttpMessageNotReadableException.class})
  5. public ResponseEntity&lt;Object&gt; handleJacksonError(final JsonMappingException ex, final WebRequest request) {
  6. log.error(&quot;Cannot parse request. {}&quot;, request.getDescription(true), ex);
  7. return new ResponseEntity&lt;&gt;(ex.getMessage(), HttpStatus.BAD_REQUEST);
  8. }
  9. }

huangapple
  • 本文由 发表于 2020年7月23日 18:33:29
  • 转载请务必保留本文链接:https://java.coder-hub.com/63052223.html
匿名

发表评论

匿名网友

:?: :razz: :sad: :evil: :!: :smile: :oops: :grin: :eek: :shock: :???: :cool: :lol: :mad: :twisted: :roll: :wink: :idea: :arrow: :neutral: :cry: :mrgreen:

确定