Nginx Lua WAF通用绕过方法

Track-聂风   ·   发表于 2018-06-07 11:21:44   ·   漏洞文章
<div class="article-content" style="margin: 30px 0px 0px; padding: 0px; box-sizing: border-box; line-height: 30px; overflow: hidden; color: rgb(88, 88, 88); font-family: Lato, "white-space:normal;";"><p id="h1-nginx-lua-waf-" style="margin-top: 0px; margin-bottom: 1rem; box-sizing: border-box;"><span style="color: rgb(29, 173, 167);"><span style="margin-right: auto; margin-left: auto; height: auto !important;"><img class="size-full wp-image-145476 aligncenter" alt="" width="800"  src="https://p5.ssl.qhimg.com/t0182b7d008c1888936.png" style="border: 0px none; box-sizing: border-box; vertical-align: middle; max-width: 100%; display: block; margin: 12px auto; height: auto !important;"/></span></span></p><blockquote style="margin-bottom: 20px; padding: 10px 20px; quotes: none; box-sizing: border-box; font-size: 14px; background: rgb(247, 247, 247); border-left: 5px solid rgba(25, 148, 145, 0.27); color: rgba(25, 148, 145, 0.973); -webkit-margin-before: 1em;"><p style="margin-top: 0px; margin-bottom: 1rem; box-sizing: border-box;">Author：JoyChou<br/>Date：20180517</p></blockquote><p style="margin-top: 0px; margin-bottom: 1rem; box-sizing: border-box;">&nbsp;</p><h2 id="h2-0" style="margin-top: 12px; margin-bottom: 0.5rem; font-size: 22px; font-family: inherit; box-sizing: border-box; line-height: 2; color: rgba(0, 0, 0, 0.85);">一&nbsp;&nbsp; 、前言</h2><p style="margin-top: 0px; margin-bottom: 1rem; box-sizing: border-box;">2018年4月3日<a class="at-link" title="<span class="label label-primary">@bre4k"</span> href="https://github.com/bre4k" style="color: rgb(29, 173, 167); box-sizing: border-box;"><span class="label label-primary">@bre4k</span></a>在群里发了一个trick。</p><blockquote style="margin-bottom: 20px; padding: 10px 20px; quotes: none; box-sizing: border-box; font-size: 14px; background: rgb(247, 247, 247); border-left: 5px solid rgba(25, 148, 145, 0.27); color: rgba(25, 148, 145, 0.973); -webkit-margin-before: 1em;"><p style="margin-top: 0px; margin-bottom: 1rem; box-sizing: border-box;">Nginx Lua获取参数时，默认获取前100个参数值，其余的将被丢弃。</p></blockquote><p style="margin-top: 0px; margin-bottom: 1rem; box-sizing: border-box;">所以，用了Nginx Lua的WAF默认都会被Bypass。</p><p style="margin-top: 0px; margin-bottom: 1rem; box-sizing: border-box;">2018年4月20日，安全客上已经有人公开了这个细节，那这篇文章也就公开了。</p><p style="margin-top: 0px; margin-bottom: 1rem; box-sizing: border-box;">&nbsp;</p><h2 id="h2-1" style="margin-top: 12px; margin-bottom: 0.5rem; font-size: 22px; font-family: inherit; box-sizing: border-box; line-height: 2; color: rgba(0, 0, 0, 0.85);">二&nbsp;&nbsp; 、原理</h2><p style="margin-top: 0px; margin-bottom: 1rem; box-sizing: border-box;"><a href="https://github.com/openresty/lua-nginx-module#ngxreqget_uri_args" style="color: rgb(29, 173, 167); box-sizing: border-box;">官方描述如下</a></p><blockquote style="margin-bottom: 20px; padding: 10px 20px; quotes: none; box-sizing: border-box; font-size: 14px; background: rgb(247, 247, 247); border-left: 5px solid rgba(25, 148, 145, 0.27); color: rgba(25, 148, 145, 0.973); -webkit-margin-before: 1em;"><p style="margin-top: 0px; margin-bottom: 1rem; box-sizing: border-box;">Note that a maximum of 100 request arguments are parsed by default (including those with the same name) and that additional request arguments are silently discarded to guard against potential denial of service attacks.</p></blockquote><p style="margin-top: 0px; margin-bottom: 1rem; box-sizing: border-box;">默认情况下最多可解析100个请求参数（包括具有相同名称的请求参数），并且会自动丢弃其他请求参数以防止潜在的拒绝服务攻击。</p><p style="margin-top: 0px; margin-bottom: 1rem; box-sizing: border-box;">搜索<code style="box-sizing: border-box; font-family: SFMono-Regular, Menlo, Monaco, Consolas, "font-size:14px;color:#E83E8C;word-break:break-word;";">100</code>，大致有下面的方法存在同样的问题。</p><ul style="margin-bottom: 1rem; padding: 0px; list-style: none; box-sizing: border-box;"><li style="padding-top: 0px; padding-right: 0px; padding-bottom: 0px; box-sizing: border-box; list-style-type: none;"><code style="box-sizing: border-box; font-family: SFMono-Regular, Menlo, Monaco, Consolas, "font-size:14px;color:#E83E8C;word-break:break-word;";">ngx.req.get_uri_args</code>&nbsp;获取get的请求参数</li><li style="padding-top: 0px; padding-right: 0px; padding-bottom: 0px; box-sizing: border-box; list-style-type: none;"><code style="box-sizing: border-box; font-family: SFMono-Regular, Menlo, Monaco, Consolas, "font-size:14px;color:#E83E8C;word-break:break-word;";">ngx.req.get_post_args</code>&nbsp;获取post的请求参数</li><li style="padding-top: 0px; padding-right: 0px; padding-bottom: 0px; box-sizing: border-box; list-style-type: none;"><code style="box-sizing: border-box; font-family: SFMono-Regular, Menlo, Monaco, Consolas, "font-size:14px;color:#E83E8C;word-break:break-word;";">ngx.req.get_headers</code>&nbsp;获取request头</li><li style="padding-top: 0px; padding-right: 0px; padding-bottom: 0px; box-sizing: border-box; list-style-type: none;"><code style="box-sizing: border-box; font-family: SFMono-Regular, Menlo, Monaco, Consolas, "font-size:14px;color:#E83E8C;word-break:break-word;";">ngx.decode_args</code>&nbsp;对参数进行URL解码</li><li style="padding-top: 0px; padding-right: 0px; padding-bottom: 0px; box-sizing: border-box; list-style-type: none;"><code style="box-sizing: border-box; font-family: SFMono-Regular, Menlo, Monaco, Consolas, "font-size:14px;color:#E83E8C;word-break:break-word;";">ngx.resp.get_headers</code>&nbsp;获取response头</li></ul><p style="margin-top: 0px; margin-bottom: 1rem; box-sizing: border-box;">在<a href="https://github.com/openresty/lua-nginx-module/blob/master/src/ngx_http_lua_common.h" style="color: rgb(29, 173, 167); box-sizing: border-box;">lua-nginx-module源码</a>里，我们可以看到源代码设置了默认的最多请求参数和头都为100</p><pre style="margin-top: 0px; margin-bottom: 0px; padding: 0px; box-sizing: border-box; font-family: SFMono-Regular, Menlo, Monaco, Consolas, "font-size:14px;margin-top:0px;margin-bottom:1rem;overflow:auto;color:#212529;"; position: relative;">#ifndef&nbsp;NGX_HTTP_LUA_MAX_ARGS&nbsp;#define&nbsp;NGX_HTTP_LUA_MAX_ARGS&nbsp;100&nbsp;#endif&nbsp;#ifndef&nbsp;NGX_HTTP_LUA_MAX_HEADERS&nbsp;#define&nbsp;NGX_HTTP_LUA_MAX_HEADERS&nbsp;100&nbsp;#endif</pre><p style="margin-top: 0px; margin-bottom: 1rem; box-sizing: border-box;">不过官方提供了方法，可修改该默认值，比如<code style="box-sizing: border-box; font-family: SFMono-Regular, Menlo, Monaco, Consolas, "font-size:14px;color:#E83E8C;word-break:break-word;";">ngx.req.get_uri_args(200)</code>就能获取前200个请求参数。</p><p style="margin-top: 0px; margin-bottom: 1rem; box-sizing: border-box;">&nbsp;</p><h2 id="h2-2" style="margin-top: 12px; margin-bottom: 0.5rem; font-size: 22px; font-family: inherit; box-sizing: border-box; line-height: 2; color: rgba(0, 0, 0, 0.85);">三 、 测试</h2><pre style="margin-top: 0px; margin-bottom: 0px; padding: 0px; box-sizing: border-box; font-family: SFMono-Regular, Menlo, Monaco, Consolas, "font-size:14px;margin-top:0px;margin-bottom:1rem;overflow:auto;color:#212529;"; position: relative;">location&nbsp;=&nbsp;/test&nbsp;{
&nbsp;&nbsp;&nbsp;&nbsp;content_by_lua_block&nbsp;{
&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;local&nbsp;args&nbsp;=&nbsp;ngx.req.get_uri_args()
&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;for&nbsp;key,&nbsp;val&nbsp;in&nbsp;pairs(args)&nbsp;do
&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;if&nbsp;type(val)&nbsp;==&nbsp;"table"&nbsp;then
&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;ngx.say(key,&nbsp;":&nbsp;",&nbsp;table.concat(val,&nbsp;",&nbsp;"))
&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;else
&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;ngx.say(key,&nbsp;":&nbsp;",&nbsp;val)
&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;if&nbsp;val&nbsp;==&nbsp;&nbsp;&#39;joychou&#39;&nbsp;then
&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;ngx.say("I&nbsp;got&nbsp;u,joychou")
&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;return
&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;end
&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;end
&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;end
&nbsp;&nbsp;&nbsp;&nbsp;}
}</pre><p style="margin-top: 0px; margin-bottom: 1rem; box-sizing: border-box;">当请求参数为101个时，此时获取不到最后一个请求参数。</p><pre style="margin-top: 0px; margin-bottom: 0px; padding: 0px; box-sizing: border-box; font-family: SFMono-Regular, Menlo, Monaco, Consolas, "font-size:14px;margin-top:0px;margin-bottom:1rem;overflow:auto;color:#212529;"; position: relative;">curl&nbsp;-v&nbsp;&#39;http://test.joychou.org/test?a1=1&a2=2&a3=3&a4=4&a5=5&a6=6&a7=7&a8=8&a9=9&a10=10&a11=11&a12=12&a13=13&a14=14&a15=15&a16=16&a17=17&a18=18&a19=19&a20=20&a21=21&a22=22&a23=23&a24=24&a25=25&a26=26&a27=27&a28=28&a29=29&a30=30&a31=31&a32=32&a33=33&a34=34&a35=35&a36=36&a37=37&a38=38&a39=39&a40=40&a41=41&a42=42&a43=43&a44=44&a45=45&a46=46&a47=47&a48=48&a49=49&a50=50&a51=51&a52=52&a53=53&a54=54&a55=55&a56=56&a57=57&a58=58&a59=59&a60=60&a61=61&a62=62&a63=63&a64=64&a65=65&a66=66&a67=67&a68=68&a69=69&a70=70&a71=71&a72=72&a73=73&a74=74&a75=75&a76=76&a77=77&a78=78&a79=79&a80=80&a81=81&a82=82&a83=83&a84=84&a85=85&a86=86&a87=87&a88=88&a89=89&a90=90&a91=91&a92=92&a93=93&a94=94&a95=95&a96=96&a97=97&a98=98&a99=99&a100=100&a=joychou&#39;</pre><p style="margin-top: 0px; margin-bottom: 1rem; box-sizing: border-box;">减少一个请求参数，当请求参数刚好为100个时，能获取到最后一个请求参数。</p><pre style="margin-top: 0px; margin-bottom: 0px; padding: 0px; box-sizing: border-box; font-family: SFMono-Regular, Menlo, Monaco, Consolas, "font-size:14px;margin-top:0px;margin-bottom:1rem;overflow:auto;color:#212529;"; position: relative;">a77:&nbsp;77&nbsp;a9:&nbsp;9&nbsp;a43:&nbsp;43&nbsp;a24:&nbsp;24&nbsp;a52:&nbsp;52&nbsp;a61:&nbsp;61&nbsp;a35:&nbsp;35&nbsp;a70:&nbsp;70&nbsp;a78:&nbsp;78&nbsp;a42:&nbsp;42&nbsp;a53:&nbsp;53&nbsp;a49:&nbsp;49&nbsp;a87:&nbsp;87&nbsp;a60:&nbsp;60&nbsp;a58:&nbsp;58&nbsp;a96:&nbsp;96&nbsp;a14:&nbsp;14&nbsp;a27:&nbsp;27&nbsp;a15:&nbsp;15&nbsp;a85:&nbsp;85&nbsp;a36:&nbsp;36&nbsp;a26:&nbsp;26&nbsp;a41:&nbsp;41&nbsp;a94:&nbsp;94&nbsp;a37:&nbsp;37&nbsp;a50:&nbsp;50&nbsp;a63:&nbsp;63&nbsp;a48:&nbsp;48&nbsp;a72:&nbsp;72&nbsp;a12:&nbsp;12&nbsp;a29:&nbsp;29&nbsp;a59:&nbsp;59&nbsp;a38:&nbsp;38&nbsp;a62:&nbsp;62&nbsp;a:&nbsp;joychou&nbsp;I&nbsp;got&nbsp;u,&nbsp;joychou</pre><p style="margin-top: 0px; margin-bottom: 1rem; box-sizing: border-box;">说明，默认确实是100个请求参数，并且100个请求参数以后的请求参数将会被丢弃。</p><p style="margin-top: 0px; margin-bottom: 1rem; box-sizing: border-box;">有一个奇怪的地方，在上面的返回内容中，在大概中间的位置，就已经输出了最后的joychou参数。我的猜测是输出并不是按照顺序，但是解析的顺序确实按照参数提交的顺序，因为当第101个参数value是joychou时，不能获取到该值。</p><p style="margin-top: 0px; margin-bottom: 1rem; box-sizing: border-box;">附上生成参数的python脚本：</p><pre style="margin-top: 0px; margin-bottom: 0px; padding: 0px; box-sizing: border-box; font-family: SFMono-Regular, Menlo, Monaco, Consolas, "font-size:14px;margin-top:0px;margin-bottom:1rem;overflow:auto;color:#212529;"; position: relative;">#&nbsp;author:&nbsp;JoyChou&nbsp;#&nbsp;mail:&nbsp;joychou<span class="label label-primary">@joychou.org</span>&nbsp;#&nbsp;date:&nbsp;2018-04-03&nbsp;a&nbsp;=&nbsp;&#39;&#39;&nbsp;for&nbsp;i&nbsp;in&nbsp;range(200):
&nbsp;&nbsp;&nbsp;&nbsp;a&nbsp;=&nbsp;a&nbsp;+&nbsp;&#39;a{0}={1}&&#39;.format(i+1,&nbsp;i+1)&nbsp;print&nbsp;a</pre><p style="margin-top: 0px; margin-bottom: 1rem; box-sizing: border-box;">&nbsp;</p><h2 id="h2-3" style="margin-top: 12px; margin-bottom: 0.5rem; font-size: 22px; font-family: inherit; box-sizing: border-box; line-height: 2; color: rgba(0, 0, 0, 0.85);">四 、 修复</h2><p style="margin-top: 0px; margin-bottom: 1rem; box-sizing: border-box;">当然，不建议在源码修改参数的MAX值。因为你设置再大的值都能被绕过。建议通过方法的参数去设置。</p><p style="margin-top: 0px; margin-bottom: 1rem; box-sizing: border-box;">如果能获取到请求参数的长度，再利用类似<code style="box-sizing: border-box; font-family: SFMono-Regular, Menlo, Monaco, Consolas, "font-size:14px;color:#E83E8C;word-break:break-word;";">ngx.req.get_uri_args(lenth)</code>方式，不是就可以了吗？</p><p style="margin-top: 0px; margin-bottom: 1rem; box-sizing: border-box;">阅读文档发现，其实并不能获取到请求参数的个数。但是如果设置lenth为0的话，就能获取所有请求参数。</p><blockquote style="margin-bottom: 20px; padding: 10px 20px; quotes: none; box-sizing: border-box; font-size: 14px; background: rgb(247, 247, 247); border-left: 5px solid rgba(25, 148, 145, 0.27); color: rgba(25, 148, 145, 0.973); -webkit-margin-before: 1em;"><p style="margin-top: 0px; margin-bottom: 1rem; box-sizing: border-box;">This argument can be set to zero to remove the limit and to process all request arguments received.</p></blockquote><pre style="margin-top: 0px; margin-bottom: 0px; padding: 0px; box-sizing: border-box; font-family: SFMono-Regular, Menlo, Monaco, Consolas, "font-size:14px;margin-top:0px;margin-bottom:1rem;overflow:auto;color:#212529;"; position: relative;">local&nbsp;args&nbsp;=&nbsp;ngx.req.get_uri_args(0)</pre><p style="margin-top: 0px; margin-bottom: 1rem; box-sizing: border-box;">不过官方强烈不建议设置为0的方式，防止潜在的拒绝服务攻击。</p><blockquote style="margin-bottom: 20px; padding: 10px 20px; quotes: none; box-sizing: border-box; font-size: 14px; background: rgb(247, 247, 247); border-left: 5px solid rgba(25, 148, 145, 0.27); color: rgba(25, 148, 145, 0.973); -webkit-margin-before: 1em;"><p style="margin-top: 0px; margin-bottom: 1rem; box-sizing: border-box;">Removing the max_args cap is strongly discouraged.</p></blockquote><p style="margin-top: 0px; margin-bottom: 1rem; box-sizing: border-box;">其实，我在想为什么设置0，就会有潜在的拒绝服务攻击，请求反正都会到nginx，无论外面get或者post的参数再多。</p><p style="margin-top: 0px; margin-bottom: 1rem; box-sizing: border-box;">我给官方提了一个<a href="https://github.com/openresty/lua-nginx-module/issues/1294" style="color: rgb(29, 173, 167); box-sizing: border-box;">Issue</a>， 作者说，<code style="box-sizing: border-box; font-family: SFMono-Regular, Menlo, Monaco, Consolas, "font-size:14px;color:#E83E8C;word-break:break-word;";">ngx.req.get_uri_args(0)</code>会增加服务端CPU和内存的使用。后来，我测试200个参数，利用Nginx+php，获取第200个参数，能获取到，说明Nginx默认不会对请求参数个数进行限制。那么可能存在问题的地方就在于Nginx Lua本身，当Nginx Lua利用<code style="box-sizing: border-box; font-family: SFMono-Regular, Menlo, Monaco, Consolas, "font-size:14px;color:#E83E8C;word-break:break-word;";">ngx.req.get_uri_args(0)</code>获取所有参数，并且进行循环遍历，一旦请求参数非常多，就会消耗更多的CPU和内存，最后甚至导致拒绝服务。</p><p style="margin-top: 0px; margin-bottom: 1rem; box-sizing: border-box;">2018年04年03日，Bypass007在官方提了一个关于这个漏洞修复的<a href="https://github.com/openresty/openresty/issues/358" style="color: rgb(29, 173, 167); box-sizing: border-box;">ISSUE</a>，最后官方在2018年04月21日，根据这个ISSUE新增了一个功能。该功能的commit记录地址：<a href="https://github.com/openresty/lua-nginx-module/commit/52af63a5b949d6da2289e2de3fb839e2aba4cbfd" style="color: rgb(29, 173, 167); box-sizing: border-box;">https://github.com/openresty/lua-nginx-module/commit/52af63a5b949d6da2289e2de3fb839e2aba4cbfd</a></p><p style="margin-top: 0px; margin-bottom: 1rem; box-sizing: border-box;">功能描述为：在v0.10.13后的版本(包括v0.10.13)，当限制的请求数被突破后，第二个返回值是<code style="box-sizing: border-box; font-family: SFMono-Regular, Menlo, Monaco, Consolas, "font-size:14px;color:#E83E8C;word-break:break-word;";">truncated</code>字符串。</p><blockquote style="margin-bottom: 20px; padding: 10px 20px; quotes: none; box-sizing: border-box; font-size: 14px; background: rgb(247, 247, 247); border-left: 5px solid rgba(25, 148, 145, 0.27); color: rgba(25, 148, 145, 0.973); -webkit-margin-before: 1em;"><p style="margin-top: 0px; margin-bottom: 1rem; box-sizing: border-box;">Since v0.10.13, when the limit is exceeded, it will return a second value which is the string “truncated”.<br/>However, the optional max_args function argument can be used to override this limit:</p></blockquote><pre style="margin-top: 0px; margin-bottom: 0px; padding: 0px; box-sizing: border-box; font-family: SFMono-Regular, Menlo, Monaco, Consolas, "font-size:14px;margin-top:0px;margin-bottom:1rem;overflow:auto;color:#212529;"; position: relative;">&nbsp;local&nbsp;args,&nbsp;err&nbsp;=&nbsp;ngx.req.get_uri_args(10)&nbsp;if&nbsp;err&nbsp;==&nbsp;"truncated"&nbsp;then&nbsp;--&nbsp;one&nbsp;can&nbsp;choose&nbsp;to&nbsp;ignore&nbsp;or&nbsp;reject&nbsp;the&nbsp;current&nbsp;request&nbsp;here&nbsp;end</pre><p style="margin-top: 0px; margin-bottom: 1rem; box-sizing: border-box;">所以，最终的修复方法出来了。</p><ul style="margin-bottom: 1rem; padding: 0px; list-style: none; box-sizing: border-box;"><li style="padding-top: 0px; padding-right: 0px; padding-bottom: 0px; box-sizing: border-box; list-style-type: none;">升级<code style="box-sizing: border-box; font-family: SFMono-Regular, Menlo, Monaco, Consolas, "font-size:14px;color:#E83E8C;word-break:break-word;";">lua-nginx-module</code>版本到v0.10.13或以上</li><li style="padding-top: 0px; padding-right: 0px; padding-bottom: 0px; box-sizing: border-box; list-style-type: none;">再限制参数总数，至于总数限制为多少，我个人觉得100个已经足够了</li></ul><p style="margin-top: 0px; margin-bottom: 1rem; box-sizing: border-box;">没有找到相关OpenResty升级模块的资料，自己鼓捣了下<code style="box-sizing: border-box; font-family: SFMono-Regular, Menlo, Monaco, Consolas, "font-size:14px;color:#E83E8C;word-break:break-word;";">OpenResty</code>如何升级<code style="box-sizing: border-box; font-family: SFMono-Regular, Menlo, Monaco, Consolas, "font-size:14px;color:#E83E8C;word-break:break-word;";">lua-nginx-module</code>模块。</p><p style="margin-top: 0px; margin-bottom: 1rem; box-sizing: border-box;">步骤如下：</p><pre style="margin-top: 0px; margin-bottom: 0px; padding: 0px; box-sizing: border-box; font-family: SFMono-Regular, Menlo, Monaco, Consolas, "font-size:14px;margin-top:0px;margin-bottom:1rem;overflow:auto;color:#212529;"; position: relative;">&nbsp;下载lua-nginx-module模块对应版本
wget&nbsp;https://github.com/openresty/lua-nginx-module/releases/tag/v0.10.13&nbsp;解压
tar&nbsp;-zxvf&nbsp;v0.10.13&nbsp;删除之前的lua-nginx-module版本
rm&nbsp;-rf&nbsp;openresty-1.9.15.1/bundle/ngx_lua-0.10.5&nbsp;复制新的lua-nginx-module&nbsp;mv&nbsp;lua-nginx-module-0.10.13&nbsp;openresty-1.9.15.1/bundle/ngx_lua-0.10.13&nbsp;编译，参数参考VeryNginx的编译参数
./configure&nbsp;--prefix=/opt/verynginx/openresty&nbsp;--user=nginx&nbsp;--group=nginx&nbsp;--with-http_v2_module&nbsp;--with-http_sub_module&nbsp;--with-http_stub_status_module&nbsp;--with-luajit

make</pre><p style="margin-top: 0px; margin-bottom: 1rem; box-sizing: border-box;">&nbsp;</p><h2 id="h2-4" style="margin-top: 12px; margin-bottom: 0.5rem; font-size: 22px; font-family: inherit; box-sizing: border-box; line-height: 2; color: rgba(0, 0, 0, 0.85);">五 、案例绕过</h2><p style="margin-top: 0px; margin-bottom: 1rem; box-sizing: border-box;">需要申明的是，下面几种WAF我不确定是否用的Nginx Lua，只是可以用参数总数的方式进行绕过而已。</p><h3 id="h3-5" style="margin-top: 12px; margin-bottom: 0.5rem; font-size: 19px; font-family: inherit; box-sizing: border-box; line-height: 2; color: rgba(0, 0, 0, 0.85);"><a></a>5.1 阿里WAF</h3><p style="margin-top: 0px; margin-bottom: 1rem; box-sizing: border-box;">此WAF是阿里内部使用的WAF，即<code style="box-sizing: border-box; font-family: SFMono-Regular, Menlo, Monaco, Consolas, "font-size:14px;color:#E83E8C;word-break:break-word;";">*.taobao.com</code>等域名使用的WAF，并未测试阿里云对外售卖的云WAF。</p><p style="margin-top: 0px; margin-bottom: 1rem; box-sizing: border-box;">先请求一个POST的XSS Payload，拦截。</p><pre style="margin-top: 0px; margin-bottom: 0px; padding: 0px; box-sizing: border-box; font-family: SFMono-Regular, Menlo, Monaco, Consolas, "font-size:14px;margin-top:0px;margin-bottom:1rem;overflow:auto;color:#212529;"; position: relative;">curl&nbsp;-v&nbsp;-d&nbsp;&#39;a=<img&nbsp;src=x&nbsp;onerror=alert(/xss/)>&#39;&nbsp;lu.taobao.com</pre><p style="margin-top: 0px; margin-bottom: 1rem; box-sizing: border-box;">通过Fuzz发现，当增加参数的个数到478后，带着XSS Payload，不再进行拦截，并且网站能正常访问。</p><pre style="margin-top: 0px; margin-bottom: 0px; padding: 0px; box-sizing: border-box; font-family: SFMono-Regular, Menlo, Monaco, Consolas, "font-size:14px;margin-top:0px;margin-bottom:1rem;overflow:auto;color:#212529;"; position: relative;">curl&nbsp;-v&nbsp;-d&nbsp;&#39;a1=1&a2=2&a3=3&a4=4&a5=5&a6=6&a7=7&a8=8&a9=9&a10=10&a11=11&a12=12&a13=13&a14=14&a15=15&a16=16&a17=17&a18=18&a19=19&a20=20&a21=21&a22=22&a23=23&a24=24&a25=25&a26=26&a27=27&a28=28&a29=29&a30=30&a31=31&a32=32&a33=33&a34=34&a35=35&a36=36&a37=37&a38=38&a39=39&a40=40&a41=41&a42=42&a43=43&a44=44&a45=45&a46=46&a47=47&a48=48&a49=49&a50=50&a51=51&a52=52&a53=53&a54=54&a55=55&a56=56&a57=57&a58=58&a59=59&a60=60&a61=61&a62=62&a63=63&a64=64&a65=65&a66=66&a67=67&a68=68&a69=69&a70=70&a71=71&a72=72&a73=73&a74=74&a75=75&a76=76&a77=77&a78=78&a79=79&a80=80&a81=81&a82=82&a83=83&a84=84&a85=85&a86=86&a87=87&a88=88&a89=89&a90=90&a91=91&a92=92&a93=93&a94=94&a95=95&a96=96&a97=97&a98=98&a99=99&a100=100&a101=101&a102=102&a103=103&a104=104&a105=105&a106=106&a107=107&a108=108&a109=109&a110=110&a111=111&a112=112&a113=113&a114=114&a115=115&a116=116&a117=117&a118=118&a119=119&a120=120&a121=121&a122=122&a123=123&a124=124&a125=125&a126=126&a127=127&a128=128&a129=129&a130=130&a131=131&a132=132&a133=133&a134=134&a135=135&a136=136&a137=137&a138=138&a139=139&a140=140&a141=141&a142=142&a143=143&a144=144&a145=145&a146=146&a147=147&a148=148&a149=149&a150=150&a151=151&a152=152&a153=153&a154=154&a155=155&a156=156&a157=157&a158=158&a159=159&a160=160&a161=161&a162=162&a163=163&a164=164&a165=165&a166=166&a167=167&a168=168&a169=169&a170=170&a171=171&a172=172&a173=173&a174=174&a175=175&a176=176&a177=177&a178=178&a179=179&a180=180&a181=181&a182=182&a183=183&a184=184&a185=185&a186=186&a187=187&a188=188&a189=189&a190=190&a191=191&a192=192&a193=193&a194=194&a195=195&a196=196&a197=197&a198=198&a199=199&a200=200&a201=201&a202=202&a203=203&a204=204&a205=205&a206=206&a207=207&a208=208&a209=209&a210=210&a211=211&a212=212&a213=213&a214=214&a215=215&a216=216&a217=217&a218=218&a219=219&a220=220&a221=221&a222=222&a223=223&a224=224&a225=225&a226=226&a227=227&a228=228&a229=229&a230=230&a231=231&a232=232&a233=233&a234=234&a235=235&a236=236&a237=237&a238=238&a239=239&a240=240&a241=241&a242=242&a243=243&a244=244&a245=245&a246=246&a247=247&a248=248&a249=249&a250=250&a251=251&a252=252&a253=253&a254=254&a255=255&a256=256&a257=257&a258=258&a259=259&a260=260&a261=261&a262=262&a263=263&a264=264&a265=265&a266=266&a267=267&a268=268&a269=269&a270=270&a271=271&a272=272&a273=273&a274=274&a275=275&a276=276&a277=277&a278=278&a279=279&a280=280&a281=281&a282=282&a283=283&a284=284&a285=285&a286=286&a287=287&a288=288&a289=289&a290=290&a291=291&a292=292&a293=293&a294=294&a295=295&a296=296&a297=297&a298=298&a299=299&a300=300&a301=301&a302=302&a303=303&a304=304&a305=305&a306=306&a307=307&a308=308&a309=309&a310=310&a311=311&a312=312&a313=313&a314=314&a315=315&a316=316&a317=317&a318=318&a319=319&a320=320&a321=321&a322=322&a323=323&a324=324&a325=325&a326=326&a327=327&a328=328&a329=329&a330=330&a331=331&a332=332&a333=333&a334=334&a335=335&a336=336&a337=337&a338=338&a339=339&a340=340&a341=341&a342=342&a343=343&a344=344&a345=345&a346=346&a347=347&a348=348&a349=349&a350=350&a351=351&a352=352&a353=353&a354=354&a355=355&a356=356&a357=357&a358=358&a359=359&a360=360&a361=361&a362=362&a363=363&a364=364&a365=365&a366=366&a367=367&a368=368&a369=369&a370=370&a371=371&a372=372&a373=373&a374=374&a375=375&a376=376&a377=377&a378=378&a379=379&a380=380&a381=381&a382=382&a383=383&a384=384&a385=385&a386=386&a387=387&a388=388&a389=389&a390=390&a391=391&a392=392&a393=393&a394=394&a395=395&a396=396&a397=397&a398=398&a399=399&a400=400&a401=401&a402=402&a403=403&a404=404&a405=405&a406=406&a407=407&a408=408&a409=409&a410=410&a411=411&a412=412&a413=413&a414=414&a415=415&a416=416&a417=417&a418=418&a419=419&a420=420&a421=421&a422=422&a423=423&a424=424&a425=425&a426=426&a427=427&a428=428&a429=429&a430=430&a431=431&a432=432&a433=433&a434=434&a435=435&a436=436&a437=437&a438=438&a439=439&a440=440&a441=441&a442=442&a443=443&a444=444&a445=445&a446=446&a447=447&a448=448&a449=449&a450=450&a451=451&a452=452&a453=453&a454=454&a455=455&a456=456&a457=457&a458=458&a459=459&a460=460&a461=461&a462=462&a463=463&a464=464&a465=465&a466=466&a467=467&a468=468&a469=469&a470=470&a471=471&a472=472&a473=473&a474=474&a475=475&a476=476&a477=477&a=<img&nbsp;src=x&nbsp;onerror=alert(/xss/)>&#39;&nbsp;lu.taobao.com</pre><h3 id="h3-6" style="margin-top: 12px; margin-bottom: 0.5rem; font-size: 19px; font-family: inherit; box-sizing: border-box; line-height: 2; color: rgba(0, 0, 0, 0.85);"><a></a>5.2 腾讯WAF</h3><p style="margin-top: 0px; margin-bottom: 1rem; box-sizing: border-box;">此WAF是腾讯内部使用的WAF，即<code style="box-sizing: border-box; font-family: SFMono-Regular, Menlo, Monaco, Consolas, "font-size:14px;color:#E83E8C;word-break:break-word;";">*.qq.com</code>等域名使用的WAF，并未测试腾讯云对外售卖的云WAF。</p><p style="margin-top: 0px; margin-bottom: 1rem; box-sizing: border-box;">当请求参数增加到4000，不会再进行拦截，并且网站能正常访问。随便测试以下域名都受影响。</p><pre style="margin-top: 0px; margin-bottom: 0px; padding: 0px; box-sizing: border-box; font-family: SFMono-Regular, Menlo, Monaco, Consolas, "font-size:14px;margin-top:0px;margin-bottom:1rem;overflow:auto;color:#212529;"; position: relative;">web.qq.com&nbsp;ke.qq.com&nbsp;auto.qq.com&nbsp;news.qq.com&nbsp;sports.qq.com&nbsp;time.qq.com</pre><p style="margin-top: 0px; margin-bottom: 1rem; box-sizing: border-box;">&nbsp;</p><h2 id="h2-7" style="margin-top: 12px; margin-bottom: 0.5rem; font-size: 22px; font-family: inherit; box-sizing: border-box; line-height: 2; color: rgba(0, 0, 0, 0.85);">六 、 总结</h2><p style="margin-top: 0px; margin-bottom: 1rem; box-sizing: border-box;">这个问题很简单，认真读文档都能发现这个问题。但是自己为什么没发现呢？我觉得还是是思考太少。</p><p style="margin-top: 0px; margin-bottom: 1rem; box-sizing: border-box;">&nbsp;</p><h2 id="h2-8" style="margin-top: 12px; margin-bottom: 0.5rem; font-size: 22px; font-family: inherit; box-sizing: border-box; line-height: 2; color: rgba(0, 0, 0, 0.85);">七&nbsp;&nbsp;&nbsp; 、Reference</h2><ul style="margin-bottom: 1rem; padding: 0px; list-style: none; box-sizing: border-box;"><li style="padding-top: 0px; padding-right: 0px; padding-bottom: 0px; box-sizing: border-box; list-style-type: none;"><a href="https://github.com/p0pr0ck5/lua-resty-waf/issues/280" style="color: rgb(29, 173, 167); box-sizing: border-box;">https://github.com/p0pr0ck5/lua-resty-waf/issues/280</a></li><li style="padding-top: 0px; padding-right: 0px; padding-bottom: 0px; box-sizing: border-box; list-style-type: none;"><a href="https://github.com/openresty/lua-nginx-module#ngxreqget_uri_args" style="color: rgb(29, 173, 167); box-sizing: border-box;">https://github.com/openresty/lua-nginx-module#ngxreqget_uri_args</a></li><li style="padding-top: 0px; padding-right: 0px; padding-bottom: 0px; box-sizing: border-box; list-style-type: none;"><a href="https://github.com/openresty/openresty/issues/358" style="color: rgb(29, 173, 167); box-sizing: border-box;">https://github.com/openresty/openresty/issues/358</a></li><li style="padding-top: 0px; padding-right: 0px; padding-bottom: 0px; box-sizing: border-box; list-style-type: none;"><a href="https://github.com/openresty/lua-nginx-module/commit/52af63a5b949d6da2289e2de3fb839e2aba4cbfd" style="color: rgb(29, 173, 167); box-sizing: border-box;">https://github.com/openresty/lua-nginx-module/commit/52af63a5b949d6da2289e2de3fb839e2aba4cbfd</a></li></ul></div><p><section><div style="margin: 0px; padding: 0px; box-sizing: border-box;"><div style="margin: 0px; padding: 0px; box-sizing: border-box;">本文由安全客原创发布 作者：美丽联合安全 SRC<br/>转载，请参考<a href="https://www.anquanke.com/note/repost" target="_blank" style="color: rgb(61, 70, 77); box-sizing: border-box;">转载声明</a>，注明出处：&nbsp;<a href="https://www.anquanke.com/post/id/145428" target="_blank" style="color: rgb(61, 70, 77); box-sizing: border-box;">https://www.anquanke.com/post/id/145428</a>&nbsp;</div></div></section></p><p><br/></p>

打赏我,让我更有动力~

0 条回复   |  直到 2018-6-7 | 1325 次浏览
登录后才可发表内容
返回顶部 投诉反馈

© 2016 - 2025 掌控者 All Rights Reserved.