翻译自:https://blog.detectify.com/2018/12/13/jquery-file-upload-a-tale-of-three-vulnerabilities/
翻译:聂心明
在github上星个数第二多的JavaScript项目中有两个远程命令执行漏洞,并且星数第三多的那个项目可能会通过jQuery-File-Upload任意删掉被上传的文件。后者是有意的行为,我们的安全研究员警告过他们,用户的隐私内容会被外部人员随意查看,这是对用户隐私及其不尊重的。Detectify Crowdsource已经在 jQuery-File-Upload找到三个安全漏洞了,并且这些漏洞已经被安全研究员提交了我们的社区中,并且我们已经在我们的漏扫工具Detectify中实现了检测。我们的研究员发现,jQuery-File-Upload广泛存在于各种平台中,并且很多都没有被正确配置。下面是CVE-2018-9206的poc:未授权任意文件上传漏洞和基于ImageTragick的远程命令执行漏洞。下面是这三个漏洞的说明
第一个漏洞在2015年的时候就已经被发现了。但是,在2018年的时候才被分配cve编号,并且通过《Thousands of Applications were vulnerable to RCE via jQuery File Upload》才被大家所熟知。jQuery-File-Upload是一个开源的文件上传组件,在github上是星数第二多的JavaScript项目,星数第一多的项目是jQuery JavaScript。CVE-2018-9206的核心问题在于服务器配置和php的组件技术,不是JavaScript的问题。当然,这个问题最后成为JavaScript的漏洞让人觉得有点不可思议,但如果这个是php的文件就不会让人感到惊讶。这个漏洞依赖于Apache服务器中.htaccess的配置。这个配置文件会限制文件的上传或者文件的执行。
# The following directives prevent the execution of script files
# in the context of the website.
# They also force the content-type application/octet-stream and
# force browsers to display a download dialog for non-image files.
SetHandler default-handler
ForceType application/octet-stream
Header set Content-Disposition attachment
# The following unsets the forced type and Content-Disposition headers
# for known image files:
<FilesMatch "(?i)\.(gif|jpe?g|png)$">
ForceType none
Header unset Content-Disposition
</FilesMatch>
<...>
上面是jQuery-File-Upload的.htaccess文件,并且存在于9.22.0版本之前,为了防范文件上传漏洞。.htaccess可以设置MIME的类型为application/octet-stream,从而让浏览器下载服务器上的文件而不是去执行它们。这就意味着Query-File-Upload 允许任意文件上传,但是不能在服务器上执行它们,因为它们信任web服务器对此做了检查。修补之后,后面的版本检查了文件的上传类型。可是问题是,在Apache的2.3.9版本中,Apache默认不再支持.htaccess了。如果没有被明确的开启的话,这个保护措施是无效的。如果使用了另一种web容器的话(比如Nginx),那么基于.htaccess文件的保护则完全无效。攻击者可以上传任意文件到服务器中,然后控制这台服务器。如果攻击者上传一个后缀为php的文件,那么就可以执行它了。
jQuery-File-Upload的第二个漏洞在黑客社区中被广为所知,这个漏洞一直没有被公开,因为CVE-2018-9206的出现,这个漏洞才被大家所关注,之后越来越多的人开始研究jQuery-File-Upload的代码库。因为代码使用了ImageMagic,攻击者可以使用GhostScript来执行任意代码(CVE-2016-3714 AKA ImageTragick)。这里有个演示视频。攻击者会把下面的GhostScript保存成后缀为PNG, GIF 或者JPG的文件,然后再把他们上传到服务器中。
%!PS
userdict /setpagedevice undef
save
legal
{ null restore } stopped { pop } if
{ legal } stopped { pop } if
restore
mark /OutputFile (%pipe%ping example.com) currentdevice putdeviceprops
服务器会执行ping example.com这个指令,注意,在不同的操作系统中GhostScript可能看着会有一些不同,但是ping指令可以运行在大多数的环境中,这样就可以利用自动化的手段来发现存在的漏洞。注意,这个漏洞是jQuery-File-Upload库使用的问题,不是代码本身的问题。
第三个也是最后一个漏洞是不安全的对象引用,或者称之为 IDOR vulnerability,一个站长报告过这个问题,但是issue中明确写着这是一种“有意的功能”,但是很多jQuery-File-Upload 用户不知道这一特性,也不知道这一特性的风险。这就是为什么:向文件上传接口发送get请求,服务器就会返回一组json数据,里面包含所有之前上传的文件。这就会暴露文件的名称,上传的路径,缩略图路径还有就是可能会造成文件的删除。返回报文就像下面这样:
{"files":[{"name":image.jpg","size":68549,"url":"http:\/\/example.com\/image.jpg","thumbnailUrl":"http:\/\/example.com\/thumbnail\/image.jpg","deleteUrl":"http:\/\/example.com\/server/php?file=image.jpg","deleteType":"DELETE"}
通过返回报文,用户就可以通过url字段的内容看到之前上传过的文件。也可以通过发送DELETE请求来删除所有的文件,发送的请求就像下面这样:
curl -X DELETE http://example.com/server/php?file=image.jpg
当我们看到网站使用jQuery-File-Upload时,就可以故意的去访问这个“有意的漏洞”了。如果这个网站是一个约会网站,用户肯定会很自然的上传自己的图片。通过发送这样的请求,我们就可以看到所有用户上传的图片了。另一个例子是如果这个网站通过上传用户的身份证或者护照来验证用户身份的话,我就可以用这样的请求来获得所有的图片。我已经联系过 Sebastian Tschan了(jQuery-File-Upload的主要维护者),并且所有的网站都发现了这样的漏洞。
最开始提到的两个漏洞已经在jQuery-file-upload后续的版本修复了。我建议大家赶紧升级到最新的版本。为了修复最后一个漏洞,你应该严格限制文件上传接口的权限(通常在server/php/index.php),所有上传的文件不应该被公开可见。你是否在你的网站上安装了jQuery-File-Upload?并且你不确定你的网站代码是否安全。那么现在你应该来试试Detectify。