java Sping Boot 2.5到2.6迁移后,当URL包含双斜杠时出现404错误

pbgvytdp  于 2024-01-05  发布在  Java
关注(0)|答案(4)|浏览(192)

以下URL用于在Sping Boot 2.5中返回200
http://localhost:8080//actuator/health
升级到2.6后,它不再工作,并返回404 Not Found
我发现了这个错误,因为项目的一些集成测试失败了(请求返回404)。然后我注意到了双斜杠,在删除它之后,问题得到了解决。
我只是想知道是什么特征导致了这种新行为?
我把2.6 release notes检查了几次,但没有任何印象。

lnvxswe2

lnvxswe21#

问题是路径匹配策略已更改。执行器路径模式不能被

  1. spring.mvc.pathmatch.matching-strategy=ant-path-matcher

字符串
解决方案是手动修复包含双斜杠的URL。

dwbf0jvd

dwbf0jvd2#

1.您是否可以使用单斜杠访问执行器端点?如http://localhost:8080/actuator/health
1.如果是,那么你是否为你的应用程序配置了Spring Security ?你可以定义一个配置,你的应用程序将允许url中的连续斜杠。

  1. @Bean
  2. HttpFirewall httpFirewall() {
  3. StrictHttpFirewall firewall = new StrictHttpFirewall();
  4. firewall.setAllowUrlEncodedDoubleSlash(true);
  5. return firewall;
  6. }

字符串

xpcnnkqh

xpcnnkqh3#

这可能是原因:基于PathPattern的Spring MVC路径匹配策略
根据注册的Spring MVC处理程序Map匹配请求路径的默认策略已从AntPathMatcher更改为PathPatternParser。
修复:

  1. spring.mvc.pathmatch.matching-strategy=ant-path-matcher

字符串

agyaoht7

agyaoht74#

在 Spring Boot 3上测试:
1.创建RequestRejectedException处理程序:

  1. @Component
  2. public class CustomRequestRejectedHandler implements RequestRejectedHandler {
  3. @Override
  4. public void handle(HttpServletRequest req, HttpServletResponse res, RequestRejectedException ex) throws IOException {
  5. req.setAttribute("custom-error-id", ex.getMessage());
  6. res.sendError(HttpStatus.BAD_REQUEST.value());
  7. }
  8. }

字符串
它将拦截RequestRejectedException,在这里您可以自定义http状态并添加一些用于异常识别的附加信息,这些信息将在接下来的步骤中使用。

  1. Spring将重定向已处理的错误到/error端点,这也是你需要实现的。创建如下内容:
  1. @Controller
  2. @RequestMapping("/error")
  3. public class CustomErrorController implements ErrorController {
  4. @RequestMapping
  5. public ResponseEntity<ErrorResponse> handleError(HttpServletRequest request) {
  6. var httpStatus = resolveHttpStatus(request);
  7. var errorResponse = new ErrorResponse(extractErrorMessage(request, httpStatus));
  8. return new ResponseEntity<>(errorResponse, httpStatus);
  9. }
  10. protected HttpStatus resolveHttpStatus(HttpServletRequest request) {
  11. Integer statusCode = (Integer) request.getAttribute(RequestDispatcher.ERROR_STATUS_CODE);
  12. if (statusCode == null) {
  13. return HttpStatus.INTERNAL_SERVER_ERROR;
  14. }
  15. return Arrays.stream(HttpStatus.values())
  16. .filter(status -> status.value() == statusCode)
  17. .findFirst()
  18. .orElse(HttpStatus.INTERNAL_SERVER_ERROR);
  19. }
  20. private String extractErrorMessage(HttpServletRequest request, HttpStatus httpStatus) {
  21. var errorMessage = (String) request.getAttribute("custom-error-id");
  22. if (!Strings.isBlank(errorMessage)) {
  23. return errorMessage;
  24. } else {
  25. return httpStatus.getReasonPhrase();
  26. }
  27. }
  28. }


1.最后将/error端点添加到spring WebSecurity配置排除列表中:

  1. @EnableWebSecurity
  2. @Configuration
  3. public class SecurityConfig {
  4. ...
  5. @Bean
  6. public SecurityFilterChain web(HttpSecurity http) throws Exception {
  7. http
  8. ...
  9. .authorizeHttpRequests(auth-> auth
  10. .requestMatchers("/error").permitAll();
  11. ...
  12. return http.build();
  13. }


1.对于集成测试:使用WebTestClient实现它们。

展开查看全部

相关问题