Java JDBC導(dǎo)致的反序列化攻擊原理解析
這篇文章主要介紹了Java JDBC導(dǎo)致的反序列化攻擊原理解析,文中通過示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友可以參考下
背景
上周BlackHat Europe 2019的議題《New Exploit Technique In Java Deserialization Attack》中提到了一個(gè)通過注入JDBC URL實(shí)現(xiàn)反序列化攻擊的場(chǎng)景,簡(jiǎn)單分析一下。
分析
首先,當(dāng)java應(yīng)用使用MySQL Connector/J(官方的JDBC驅(qū)動(dòng),本文基于其8.0+版本)連接mysql時(shí),JDBC URL的格式如下:protocol//[hosts]/[database]?properties,具體可看mysql官方文檔,示例:jdbc:mysql://localhost:3306/test?useSSL=true
其中,protocol、host、database都比較好理解,URL中的properties可以設(shè)定MySQL Connector/J連接mysql服務(wù)器的具體方式,關(guān)于properties的官方文檔地址,其中和本文相關(guān)的連接屬性有兩個(gè),分別是autoDeserialize和queryInterceptors,前者是設(shè)定MySQL Connector/J是否反序列化BLOB類型的數(shù)據(jù),后者是攔截器,在查詢執(zhí)行時(shí)觸發(fā),由com.mysql.cj.protocol.a.NativeProtocol#sendQueryPacket方法源碼可知,會(huì)在執(zhí)行查詢語句前后分別調(diào)用攔截器的preProcess和postProcess方法。
接下來定位下反序列化的觸發(fā)點(diǎn),在mysql-connector-java組件下全局搜索關(guān)鍵字“.readObject()”,定位到com.mysql.cj.jdbc.result.ResultSetImpl類中的getObject(int columnIndex)方法,部分核心代碼如下:
public Object getObject(int columnIndex) throws SQLException {
……
case BLOB:
byte[] data = getBytes(columnIndex);
if (this.connection.getPropertySet().getBooleanProperty(PropertyDefinitions.PNAME_autoDeserialize).getValue()) {
Object obj = data;
// Serialized object?
try {
ByteArrayInputStream bytesIn = new ByteArrayInputStream(data);
ObjectInputStream objIn = new ObjectInputStream(bytesIn);
obj = objIn.readObject();
}
}
}
變量data即為mysql返回結(jié)果集,當(dāng)JDBC URL中設(shè)定屬性autoDeserialize為true時(shí),會(huì)對(duì)類型為bit、binary以及blob的數(shù)據(jù)進(jìn)行反序列化,如何觸發(fā)getObject(int columnIndex)方法的調(diào)用呢?議題中給出的調(diào)用鏈如下:
> com.mysql.cj.jdbc.interceptors.ServerStatusDiffInterceptor#preProcess/postProcess > com.mysql.cj.jdbc.interceptors.ServerStatusDiffInterceptor#populateMapWithSessionStatusValues > com.mysql.cj.jdbc.util.ResultSetUtil#resultSetToMap > com.mysql.cj.jdbc.result.ResultSetImpl#getObject
ServerStatusDiffInterceptor即為此前提到過的攔截器,在JDBC URL中設(shè)定屬性queryInterceptors為ServerStatusDiffInterceptor時(shí),執(zhí)行查詢語句會(huì)調(diào)用攔截器的preProcess和postProcess方法,進(jìn)而通過上述調(diào)用鏈最終調(diào)用getObject(int columnIndex)方法。
實(shí)際利用還有一個(gè)問題,最終調(diào)用getObject方法的對(duì)象是數(shù)據(jù)庫返回的結(jié)果集,由populateMapWithSessionStatusValues方法可知:
try {
toPopulate.clear();
stmt = this.connection.createStatement();
rs = stmt.executeQuery("SHOW SESSION STATUS");
ResultSetUtil.resultSetToMap(toPopulate, rs);
}
這個(gè)結(jié)果集是執(zhí)行SQL語句“SHOW SESSION STATUS”后數(shù)據(jù)庫返回的值,SQL語句“SHOW SESSION STATUS”返回當(dāng)前數(shù)據(jù)庫連接的狀態(tài)值,實(shí)際是讀取系統(tǒng)表INFORMATION_SCHEMA.SESSION_VARIABLES的值,也可能是PERFORMANCE_SCHEMA.SESSION_VARIABLES(Mysql版本差異導(dǎo)致)。但是mysql中INFORMATION_SCHEMA和PERFORMANCE_SCHEMA都是不允許被修改的,所以需要想辦法操縱返回的數(shù)據(jù)。
利用條件
1.本質(zhì)上還是Java原生的反序列化利用,所以需要環(huán)境中有可用的Gadget;
2.需要能偽造相關(guān)系統(tǒng)表的數(shù)據(jù),將“SHOW SESSION STATUS”的執(zhí)行結(jié)果設(shè)置為我們精心構(gòu)造的反序列化數(shù)據(jù),或者基于mysql連接協(xié)議,自定義返回?cái)?shù)據(jù),后面有時(shí)間的時(shí)候會(huì)寫寫這塊兒。
3.可控的JDBC URL
以上就是本文的全部內(nèi)容,希望對(duì)大家的學(xué)習(xí)有所幫助,也希望大家多多支持腳本之家。
相關(guān)文章
springboot實(shí)現(xiàn)文件上傳和下載功能
這篇文章主要為大家詳細(xì)介紹了springboot實(shí)現(xiàn)文件上傳和下載功能,文中示例代碼介紹的非常詳細(xì),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2017-06-06
Java線程通信中關(guān)于生產(chǎn)者與消費(fèi)者案例分析
這篇文章主要介紹了Java線程通信中關(guān)于生產(chǎn)者與消費(fèi)者案例,本文給大家介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或工作具有一定的參考借鑒價(jià)值,需要的朋友可以參考下2021-09-09
SpringBoot創(chuàng)建定時(shí)任務(wù)的示例詳解
在Spring Boot中創(chuàng)建定時(shí)任務(wù),通常使用@Scheduled注解,這是Spring框架提供的一個(gè)功能,允許你按照固定的頻率(如每天、每小時(shí)、每分鐘等)執(zhí)行某個(gè)方法,本文給大家介紹了SpringBoot創(chuàng)建定時(shí)任務(wù)的示例,需要的朋友可以參考下2024-04-04
java中的數(shù)學(xué)計(jì)算函數(shù)的總結(jié)
這篇文章主要介紹了java中的數(shù)學(xué)計(jì)算函數(shù)的總結(jié)的相關(guān)資料,需要的朋友可以參考下2017-07-07
java SelectableChannel的使實(shí)例用法講解
在本篇文章里小編給大家整理的是一篇關(guān)于java SelectableChannel的使實(shí)例用法講解內(nèi)容,有興趣的朋友們可以學(xué)習(xí)下。2021-03-03
SpringBoot登錄驗(yàn)證碼實(shí)現(xiàn)過程詳解
這篇文章主要介紹了SpringBoot登錄驗(yàn)證碼實(shí)現(xiàn)過程詳解,文中通過示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友可以參考下2020-04-04
RabbitMQ開啟SSL與SpringBoot連接測(cè)試的配置方法
本文基于 CentOS 7 + Git + OpenSSL + yum 安裝的 RabbitMQ,需要讀者提交安裝好。其他方式也可變通參考本文。對(duì)RabbitMQ開啟SSL與SpringBoot連接測(cè)試相關(guān)知識(shí)感興趣的朋友一起看看吧2022-01-01

