为了账号安全,请及时绑定邮箱和手机立即绑定

SpringBoot Whitelabel Error Page的根本原因,三种解决方案以及其特点

标签:
SpringBoot

0、简述

在学习这个学习笔记之前最好能够对spring mvc以及Tomcat有些了解,这样理解起来更加方便,如果需要知道最直接的解决方案,拖到最底部看样例代码即可。

介绍了springboot的白页出现的真正原因,主要是没有合适的匹配情况出现404情况,然后跳转到系统默认的第一个ErrorPage,也就是白页内容上,然后根据其特定分别从三个角度,1、拦截器,2、新ErrorPage,3、自定义/error路由 去解决该问题,并且介绍各自方法的优缺点,其中还有介绍到循环页面错误的本质原因等情况

1、Whitelabel Error Page 白页

什么叫Whitelabel Error Page(也叫白页),就是SpringBoot中HTTP请求出现异常的说明页,如下图


webp

image

白页内容会展示状态码、path、以及错误原因等情况,但是真正发布在线上生成环境一般不允许出现这样的情况,更多的是自定义的404页面或者500页面等。

那么现在我们就来了解下什么情况会产生白页的情况,以及如何解决这种情况。我们就以404的情况去了解其原因。

直接来到DispatcherServlet类的protected void doDispatch(HttpServletRequest request, HttpServletResponse response) throws Exception方法,其中包含的代码片段

mappedHandler = getHandler(processedRequest);// 找到合适的请求处理器if (mappedHandler == null || mappedHandler.getHandler() == null) {    // 原则上如果没有找到则会进入到这里,并且设置response的状态码为404
    // 但是经过调试并没有进入到这里
    noHandlerFound(processedRequest, response);    return;
}

在getHandler方法中会遍历当前web容器中的HandlerMapping,找出合适的处理器Handler


webp

image


webp

image

由上图可以很明显的知道便利出的当前Handler是SimpleUrlHandlerMapping,因为其中的url中包含了/**,所有的url都可以被匹配出,不会进入到后面的noHandlerFound中,适配处理器HandlerAdapter是HttpRequestHandlerAdapter实例化的对象

在mv = ha.handle(processedRequest, response, mappedHandler.getHandler())中没法找到对应的resource,设置response的状态码为404,具体可看ResourceHttpRequestHandler类的handleRequest方法

现在就相当于该请求设置了状态码为404,其他并没有做什么,mv也是为null的

这时候需要回到Tomcat的调用流程内,如果对Tomcat的调用流程请求的同学应该知道,Tomcat在连接器收到Socket套接字请求包装成为request、response等信息交由Engine->Host 等组件一层一层传递,再由各个组件的Pipeline管道接收,后续各自的Valve(阀门)一层一层的过滤处理。

这个时候来到StandardHostValve类的private void status(Request request, Response response)方法

private void status(Request request, Response response) {        int statusCode = response.getStatus();        // 查看当前状态码,当前样例是404
        // 获取当前上下文
        Context context = request.getContext();        if (context == null) {            return;
        }        if (!response.isError()) {             // 当前请求没错误
             // 是一个原子类AtomicInteger errorState,如果大于0则认为遇到错误
             
            return;
        }

        ErrorPage errorPage = context.findErrorPage(statusCode);        // 这个地方以后再说,就是解决方案的一种,设置错误页
        if (errorPage == null) {            // Look for a default error page
            errorPage = context.findErrorPage(0);
        }        if (errorPage != null && response.isErrorReportRequired()) {
            ...

webp

image

结合代码和图,再看白页清楚的写着This application has no explicit mapping for /error,路由为\error,这个缘由就是来自这里,然后进行forward跳转,路由地址是\error

webp

image

后面使用了SpringBoot提供的白页mv,渲染生产我们所看到的白页页面内容

至此,整个的流程也已经执行完成,总结一下就是请求一个不存在的链接,被发现是404请求之后转发到/error的请求上

那么解决方案就很简单了,有三种方案,不过这三种方案均是从不同的角度去解决该问题

  • 添加拦截器

  • 添加ErrorPage

  • 添加/error 路径

2、解决白页问题

2.1、添加拦截器

public class CustomHandlerInterceptor implements HandlerInterceptor {    @Override
    public boolean preHandle(HttpServletRequest request, HttpServletResponse response,
            Object handler) throws Exception {            // 一定得为true,否则拦截器就无法生效了
            // 当然可以随意各种对url的拦截处理
        return true;
    }    @Override
    public void postHandle(HttpServletRequest request, HttpServletResponse response, Object handler,
            ModelAndView modelAndView) throws Exception {            if (modelAndView != null) {              // 防止出现空指针
              // 在springboot中如果是错误页肯定不会出现mv为null的情况
              modelAndView.setViewName("/err");              // 注意:该请求只是测试试用,并没有实际的意义
           }
    }    @Override
    public void afterCompletion(HttpServletRequest request, HttpServletResponse response,
            Object handler, Exception ex) throws Exception {
    }
}@Beanpublic WebMvcConfigurerAdapter customMvcConfigurerAdapter (){    return new WebMvcConfigurerAdapter() {        @Override
        public void addInterceptors(InterceptorRegistry registry) {
            registry.addInterceptor(new CustomHandlerInterceptor()).addPathPatterns("/**");            // 添加拦截器
            super.addInterceptors(registry);
        }
    };
}

拦截器在捕获到/error 的请求之后,强制修改mv,使得最后渲染试用的mv是我们自定义设置的,而不是白页内容,其中白页本身的mv会经过ContentNegotiating视图解析器处理成为ErrorMvcAutoConfiguration$WhitelabelErrorViewConfiguration

注意这整个过期其实经过3次HTTP请求处理的,如下图是使用HTTP事件监听打印出的日志信息

webp

image


经历了/abc==>跳转/err==>跳转/error(并不显示该内容,因为发送到浏览器的内容已经经过/err 渲染完成

其中真正的调用处理流程是/abc 没有发现合适的handler,后选择交由/error 路径处理,只是后面又被拦截器拦截处理,转发给了/err 处理

缺点:会拦截该路由的所有请求,包含静态资源文件,对纯提供接口的后端服务没太多影响,其他的服务会有影响的

2.2、添加ErrorPage

添加合适的ErrorPage就不会出现跳转到拦截器默认的/error 路径,而是跳转到自定义的ErrorPage上,这点在上面的status方法已经介绍其原因了

@Beanpublic EmbeddedServletContainerCustomizer containerCustomizer() {    return new EmbeddedServletContainerCustomizer() {        @Override
        public void customize(ConfigurableEmbeddedServletContainer configurableEmbeddedServletContainer) {
            ErrorPage errorPage400 = new ErrorPage(HttpStatus.BAD_REQUEST,"/400");
            ErrorPage errorPage404 = new ErrorPage(HttpStatus.NOT_FOUND,"/404");
            ErrorPage errorPage500 = new ErrorPage(HttpStatus.INTERNAL_SERVER_ERROR,"/500");
            configurableEmbeddedServletContainer.addErrorPages(errorPage400,errorPage404,errorPage500);
        }
    };
}// ======= 分割线@ApiOperation("404请求")@GetMapping("404")public String e404() {    return "404";
}

上述代码添加了几个错误页ErrorPage跳转的路径以及其对应的HTTP错误码,我们当前的样例肯定是跳转到/404连接上去了,再执行怎么又报错了,原则上来说应该显示404.html的内容文件,同时显示的是经典的Tomcat错误页了,如下图页面展示以及日志输出的内容

webp

image


webp

image

这个就是循环跳转的问题

当系统没有指定明确的视图解析器之后,系统便会使用自带的默认解析器InternalResourceView,在渲染前会去校验当前url的问题,如果发现请求的url和目的url是一致的情况,就会认定转发自身,出现Circular view path的问题

protected String prepareForRendering(HttpServletRequest request, HttpServletResponse response)
        throws Exception {

    String path = getUrl();    if (this.preventDispatchLoop) {
        String uri = request.getRequestURI();        if (path.startsWith("/") ? uri.equals(path) : uri.equals(StringUtils.applyRelativePath(uri, path))) {            throw new ServletException("Circular view path [" + path + "]: would dispatch back " +                    "to the current handler URL [" + uri + "] again. Check your ViewResolver setup! " +                    "(Hint: This may be the result of an unspecified view, due to default view name generation.)");
        }
    }    return path;
}

那么如何解决呢,肯定得从根本目的出发

  • 添加模板解析器,这样就不会使用默认解析器了

  • 修改跳转路径

具体的解决方案就自行了解,本文不使用模板引擎渲染,直接展示body数据

@RequestMapping("/")@RestControllerpublic class ErrorController {    @ApiOperation("404请求")    @GetMapping("404")    public String e404() {
        System.out.println("404............");        return "这真的是一个404页面,你看看";
    }



作者:jwfy
链接:https://www.jianshu.com/p/b06584591086


点击查看更多内容
1人点赞

若觉得本文不错,就分享一下吧!

评论

作者其他优质文章

正在加载中
感谢您的支持,我会继续努力的~
扫码打赏,你说多少就多少
赞赏金额会直接到老师账户
支付方式
打开微信扫一扫,即可进行扫码打赏哦
今天注册有机会得

100积分直接送

付费专栏免费学

大额优惠券免费领

立即参与 放弃机会
意见反馈 帮助中心 APP下载
官方微信

举报

0/150
提交
取消