漏洞挖掘之 Spring Cloud 注入漏洞
漏洞描述
Spring 框架为现代基于 java 的企业应用程序(在任何类型的部署平台上)提供了一个全面的编程和配置模型。
Spring Cloud 中的 serveless 框架 Spring Cloud Function 中的 RoutingFunction 类的 apply 方法将请求头中的“spring.cloud.function.routing-expression”参数作为 Spel 表达式进行处理,造成 Spel 表达式注入,攻击者可通过该漏洞执行任意代码。
利用条件
3.0.0.RELEASE <= Spring Cloud Function <= 3.2.2
环境搭建
在官方网页新建一个 Spring boot 项目(https://start.spring.io/)、使用 idea 启动。
data:image/s3,"s3://crabby-images/f1727/f1727fffe22a38284fa3b372f8395cc91bcc5c97" alt=""
修改 pom.xml 配置文件
最后访问 http://127.0.0.1:8080. 出现以下页面表示成功。
【一>所有资源获取<一】1、很多已经买不到的绝版电子书 2、安全大厂内部的培训资料 3、全套工具包 4、100 份 src 源码技术文档 5、网络安全基础入门、Linux、web 安全、攻防方面的视频 6、应急响应笔记 7、 网络安全学习路线 8、ctf 夺旗赛解析 9、WEB 安全入门笔记
data:image/s3,"s3://crabby-images/ddbad/ddbade5cf98b6f8f5df30f7891a7d448da8b6510" alt=""
漏洞复现
漏洞原理
apply 方法会将 http 头部中的 Spel 表达式进行解析,从而造成 Spel 表达式注入。
查看官方diff
data:image/s3,"s3://crabby-images/759e3/759e3142c824abc2b8c7e9cb82b6a959ffa30021" alt=""
data:image/s3,"s3://crabby-images/e132a/e132a0ae2c311d0cffcc358e8488dd3e6deeafec" alt=""
进入 springframework/cloud/function/context/config/RoutingFunction 文件。进入调试模式、将断点添加到 apply()方法。
data:image/s3,"s3://crabby-images/e759a/e759adb1a34945ae676ef378fb1808d3b7c0e571" alt=""
进入到 apply()方法后、会调用 route() 在该方法中会去判读 input 是否为 message 的实例,function 是否为空、然后进入 else if 去获取头信息、获取 key 值 spring.cloud.function.routing-expression 、在中间会对有无空格做判断。然后继续向下走。
data:image/s3,"s3://crabby-images/25b7b/25b7b323e9a60b7a0f3e446afb340047e55610fd" alt=""
会进入到 springframework/cloud/function/context/config/RoutingFunction/functionFromExpression()方法。
data:image/s3,"s3://crabby-images/91969/919691f01f642649fa84626d7cc2101442c67bc2" alt=""
routingExpression 会做为参数传入到 springframework/expression/common/TemplateAwareExpressionParser/parseExpression()方法中。
data:image/s3,"s3://crabby-images/9c46a/9c46a12176aeac72a59df590c61dcc47c411d0c9" alt=""
判读其 context 是否为 none 值 在进入
springframework/expression/spel/standard/SpelExpressionParser/doPareExpression() 会 new 一个 InternalSpelExpressionParser 类调用 doPareExpression() 继续跟进。
data:image/s3,"s3://crabby-images/dcf7c/dcf7cc31f3db919dc911a1f6794c76eb5e244726" alt=""
在 springframeworl/expression/spel/stand/InternalSpelExpressionParser/doParseExpression()方法中、会在 tokenizer.process()中 对 token 进行 源码与字节码的判断操作、继续向下。
data:image/s3,"s3://crabby-images/735e6/735e6cc84e2e2bf7fad80e9c5402d990cc947d18" alt=""
会 new 一个 SpelExpression() 跟进到
springframwork/expression/spel/standard/SpelExpression/SpelExpression()。
data:image/s3,"s3://crabby-images/a386b/a386b3a794ae906b928dfcceeeac438a63136dc1" alt=""
在SpelExpression()
方法中会将将表达式赋值到this.expression
继续跟进 return 到 springframework/expression/spel/standard/SpelpressionParser/doParseExpression()、
继续 return 到springframework/expression/common/TemplateAwareExpressionPareser/pareExpression()、return
springframework/cloud/function/context/config/RoutingFunction/functionFromExpression()
data:image/s3,"s3://crabby-images/524a2/524a2f63ae9f4bcc8cfdf7f2adf920ee77cf8eb1" alt=""
在 functionFromExpression()方法中会进入 MessageUtils.toCaseInsensitiveHeadersStructure()。
data:image/s3,"s3://crabby-images/b9917/b99176eb79b59e652d57f8d99512060a3b4b66a2" alt=""
调用 MessageStructureWithCaseInsensitiveHeaderKeys(),进入到 putAll()方法 获取 message 中头信息。
data:image/s3,"s3://crabby-images/f6097/f60973717caa690838426f143eb4750d05a6af70" alt=""
最后会进入漏洞触发点。
data:image/s3,"s3://crabby-images/b55a8/b55a897cf53a4d35c4b5da6fc502202bf1b9a5ef" alt=""
漏洞测试
Payload 的构造可以参考官方测试用例。
data:image/s3,"s3://crabby-images/f804d/f804d4b9919855553de06a2648244e8816473561" alt=""
本次利用创建文件测试。使用 payload touch /tmp/xxxxxx.test.test。
data:image/s3,"s3://crabby-images/f1478/f14787337d817f0fdeb600474bdf70544dd699bc" alt=""
data:image/s3,"s3://crabby-images/72c6b/72c6b2328370f75bf01d55234d40d448f08cbeb5" alt=""
评论