Apache Shiro权限绕过漏洞分析(CVE-2020-11989)
Apache Shiro作为常用的Java安全框架,拥有执行身份验证、授权、密码和会话管理等功能,通常会和Spring等框架一起搭配使用来开发Web应用。最近研究Shiro本来是为SCTF出题做准备,但在测试过程中却发现了一些新的缺陷能导致权限绕过,便报告给Apache Shiro官方。玄武实验室安全研究人员也单独发现了另外一种绕过方式。
影响范围
- Apache Shiro < 1.5.3
- Spring框架中使用Shiro鉴权
漏洞复现
测试Demo: https://github.com/l3yx/springboot-shiro
权限配置如下,其中/admin下的路由需要登录才能访问
1 |
|
maven打包项目为test.war,部署于Tomcat。该漏洞成功利用存在下面两个条件
- 应用不能部署在根目录,也就是需要context-path,server.servlet.context-path=/test,如果为根目录则context-path为空,就会被CVE-2020-1957的patch将URL格式化,值得注意的是若Shiro版本小于1.5.2的话那么该条件就不需要。
- Spring控制器中没有另外的权限校验代码
如果直接访问 /test/admin/page
,会返回302跳转要求登录
data:image/s3,"s3://crabby-images/719e3/719e31a124dcab289f5c291aec83f14f69a7d8ec" alt="image-20200628233813192"
但是访问 /;/test/admin/page
, 就能直接绕过Shiro权限验证,访问到/admin路由中的信息
data:image/s3,"s3://crabby-images/736cf/736cf786e88a97dae7eb2cd542038b549f24ab94" alt="image-20200628234200852"
漏洞分析
由于Shiro的权限校验是通过判断url匹配来做的,如果能找到Shiro获取的url与Web框架处理url不一致的情况就能造成权限绕过。Shiro中对于URL的获取及匹配在org.apache.shiro.web.filter.mgt.PathMatchingFilterChainResolver#getChain
以访问/;/test/admin/page
举例,通过getPathWithinApplication函数得到的路径为/
data:image/s3,"s3://crabby-images/03f3b/03f3bf6fd9394b424d82f5282c75a39ae5c25362" alt="image-20200629000331817"
跟入该函数的处理逻辑
org.apache.shiro.web.util.WebUtils#getPathWithinApplication
data:image/s3,"s3://crabby-images/e7e2a/e7e2a6a5ee91a6e9fca4e47e80bb82bbf9275e67" alt="image-20200629002023207"
可以看到 org.apache.shiro.web.util.WebUtils#getRequestUri 获取到的是/
data:image/s3,"s3://crabby-images/65910/6591019571cec10afc122ee1ffd5411b34d64d61" alt="image-20200629002427590"
这里分别通过getContextPath()
getServletPath()
getPathInfo()
获取并拼接得到/;/test//admin/page
,传入后decodeAndCleanUriString变成了/
,
org.apache.shiro.web.util.WebUtils#decodeAndCleanUriString
data:image/s3,"s3://crabby-images/b8ee5/b8ee5f1c7fc5340d71bb2c958a5182e732319bc0" alt="image-20200629002813245"
在decodeAndCleanUriString,会根据ascii为59的字符也就是;
进行URL的截断,所以最终返回了/
回到最开始的/;/test/admin/page
请求,该request请求会进入spring中,spring处理url函数如下
org.springframework.web.util.UrlPathHelper#getPathWithinServletMapping
data:image/s3,"s3://crabby-images/5728c/5728cd583f05373bbe51c6aaf7802b4f58f20ce2" alt="image-20200629000952659"
data:image/s3,"s3://crabby-images/fc582/fc58243d5d4d730175a164121fa7fd6d07d89179" alt="image-20200629001017255"
在getPathWithinApplication处理下是能正确获取到context-path与路由,最终经过getPathWithinServletMapping函数格式化处理后,得到最终路径为/admin/page
,所以我们可以正常访问到该页面
data:image/s3,"s3://crabby-images/736cf/736cf786e88a97dae7eb2cd542038b549f24ab94" alt="image-20200628234200852"
因此总结来说就是当URL进入到Tomcat时,Tomcat判断/;test/admin/page
为test应用下的/admin/page路由,进入到Shiro时被;
截断被认作为/
,再进入Spring时又被正确处理为test应用下的/admin/page路由,最后导致shiro的权限绕过。
漏洞修复
Shiro 1.5.3修改了URL获取的逻辑,不单独处理context-path,具体代码如下所示
org.apache.shiro.web.util.WebUtils#getPathWithinApplication
data:image/s3,"s3://crabby-images/8f092/8f092c597838e6f17d14c60a53b3b10f2c3905a3" alt="image-20200629005615074"
因此就无法再通过构造context-path的方式来进行绕过了。
处理时间线
- 2020-6-18 16:30 边界无限安全研究员淚笑向 Apache Shiro 官方报告漏洞
- 2020-6-19 00:04 Apache Shiro 开始处理漏洞,issue为SHIRO-782
- 2020-6-22 22:49 Apache Shiro 发布致谢