博客
关于我
强烈建议你试试无所不能的chatGPT,快点击我
深入Spring Boot--使用Arthas排查应用404/401问题
阅读量:6704 次
发布时间:2019-06-25

本文共 3698 字,大约阅读时间需要 12 分钟。

背景

在Java Web/Spring Boot开发时,很常见的问题是:

  • 网页访问404了,为什么访问不到?
  • 登陆失败了,请求返回401,到底是哪个Filter拦截了我的请求?

碰到这种问题时,通常很头痛,特别是在线上环境时。

本文介绍使用Alibaba开源的Java诊断利器Arthas,来快速定位这类Web请求404/401问题。

Java Web里一个请求被处理的流程

在进入正题之前,先温习下知识。一个普通的Java Web请求处理流程大概是这样子的:

Request  -> Filter1 -> Filter2 ... -> Servlet                                        |Response <- Filter1 <- Filter2 ... <- Servlet复制代码

Demo

本文的介绍基于一个很简单的Demo:

  • 访问 ,返回200,正常打印Welconme信息
  • 访问 ,返回404
  • 访问 ,返回401

是哪个Servlet返回了404?

Demo启动后,访问: ,返回404:

$ curl http://localhost:8080/a.txt{
"timestamp":1546790485831,"status":404,"error":"Not Found","message":"No message available","path":"/a.txt"}复制代码

我们知道一个HTTP Request,大部分情况下都是由一个Servlet处理的,那么到底是哪个Servlet返回了404?

我们使用Arthas的trace命令来定位:

$ trace javax.servlet.Servlet *Press Ctrl+C to abort.Affect(class-cnt:7 , method-cnt:185) cost in 1018 ms.复制代码

然后访问 ,Arthas会打印出整个请求树,完整的输出太长,这里只截取关键的一输出:

+---[13.087083ms] org.springframework.web.servlet.DispatcherServlet:resolveViewName()|   `---[13.020984ms] org.springframework.web.servlet.DispatcherServlet:resolveViewName()|       +---[0.002777ms] java.util.List:iterator()|       +---[0.001955ms] java.util.Iterator:hasNext()|       +---[0.001739ms] java.util.Iterator:next()|       `---[12.891979ms] org.springframework.web.servlet.ViewResolver:resolveViewName()|           +---[0.089811ms] javax.servlet.GenericServlet:
()| +---[min=0.037696ms,max=0.054478ms,total=0.092174ms,count=2] org.springframework.web.servlet.view.freemarker.FreeMarkerView$GenericServletAdapter:
()复制代码

可以看出请求经过Spring MVC的DispatcherServlet处理,最终由ViewResolver分派给FreeMarkerView$GenericServletAdapter处理。所以我们可以知道这个请求最终是被FreeMarker处理的。 后面再排查FreeMarker的配置就可以了。

这个神奇的trace javax.servlet.Servlet *到底是怎样工作的呢?

实际上Arthas会匹配到JVM里所有实现了javax.servlet.Servlet的类,然后trace它们的所有函数,所以HTTP请求会被打印出来。

这里留一个小问题,为什么只访问了http://localhost:8080/a.txt,但Arthas的trace命令打印出了两个请求树?

是哪个Filter返回了401?

在Demo里,访问 ,会返回401,即没有权限。那么是哪个Filter拦截了请求?

$ curl http://localhost:8080/admin{
"timestamp":1546794743674,"status":401,"error":"Unauthorized","message":"admin filter error.","path":"/admin"}复制代码

我们还是使用Arthas的trace命令来定位,不过这次trace的是javax.servlet.Filter

$ trace javax.servlet.Filter *Press Ctrl+C to abort.Affect(class-cnt:13 , method-cnt:75) cost in 278 ms.复制代码

再次访问admin,在Arthas里,把整个请求经过哪些Filter处理,都打印为树。这里截取关键部分:

+---[0.704625ms] org.springframework.web.filter.OncePerRequestFilter:doFilterInternal()|   `---[0.60387ms] org.springframework.web.filter.RequestContextFilter:doFilterInternal()|       +---[0.022704ms] org.springframework.web.context.request.ServletRequestAttributes:
()| +---[0.217636ms] org.springframework.web.filter.RequestContextFilter:initContextHolders()| | `---[0.180323ms] org.springframework.web.filter.RequestContextFilter:initContextHolders()| | +---[0.034656ms] javax.servlet.http.HttpServletRequest:getLocale()| | +---[0.0311ms] org.springframework.context.i18n.LocaleContextHolder:setLocale()| | +---[0.008691ms] org.springframework.web.context.request.RequestContextHolder:setRequestAttributes()| | `---[0.014918ms] org.apache.commons.logging.Log:isDebugEnabled()| +---[0.215481ms] javax.servlet.FilterChain:doFilter()| | `---[0.072186ms] com.example.demo404401.AdminFilterConfig$AdminFilter:doFilter()| | `---[0.021945ms] javax.servlet.http.HttpServletResponse:sendError()复制代码

可以看到HTTP Request最终是被com.example.demo404401.AdminFilterConfig$AdminFilter处理的。

总结

  • 通过trace Servlet/Filter,可以快速定位Java Web问题
  • trace是了解应用执行流程的利器,只要trace到关键的接口或者类上
  • 仔细观察trace的结果,可以学习到Spring MVC是处理Web请求细节

链接

打个广告,Arthas正在征集使用者,您的使用是对我们最大的支持。 如果Arthas对您排查问题有帮助,请到这个Issue登记下,并在30分钟内成为Arthas Contributor:

转载地址:http://mublo.baihongyu.com/

你可能感兴趣的文章
PE文件结构学习
查看>>
[翻译] DDExpandableButton
查看>>
pthread_setschedparam
查看>>
在对listctrl的控件进行重载的过程中,GetHeaderCtrl()返回NULL的问题
查看>>
WEB网站前端性能分析相关
查看>>
sql server2008系统表详细说明sys.开头的表
查看>>
Python基础(9)--正则表达式
查看>>
解决Installation error: INSTALL_FAILED_VERSION_DOWNGRADE错误
查看>>
os 计算机的启动
查看>>
C++Vector使用方法
查看>>
字符串逆序输出
查看>>
[LeetCode] Length of Last Word 求末尾单词的长度
查看>>
[PHP100]留言板(一)
查看>>
boost::asio实现一个echo服务器
查看>>
标准差(standard deviation)和标准误差(standard error)你能解释清楚吗?
查看>>
Javascript 学习 笔记一
查看>>
写给自己看的小设计3 - 对象设计通用原则之核心原则
查看>>
Android学习笔记(四十):Preference的使用
查看>>
postgresql 修改字段名称
查看>>
c语言中的位移位操作
查看>>