siunam's Website

My personal website

Home Writeups Research Blog Projects About

Blogodogo 1/2

Table of Contents

  1. Overview
  2. Background
  3. Enumeration

Overview

Background

Try to see the content of the secret note of the administator user.

You can deploy an instance on : https://deploy.heroctf.fr
Format : Hero{flag}
Author : xanhacks

Enumeration

Home page:

A typical blog page.

In here, we can see the admin user:

And he has a secret blog post!

When we click one of those posts, we can report a post:

In this challenge, we can download a file:

┌[siunam♥earth]-(~/ctf/HeroCTF-v5/Web/Blogodogo-1-2)-[2023.05.13|19:59:40(HKT)]
└> file blog.zip 
blog.zip: Zip archive data, at least v2.0 to extract, compression method=store
┌[siunam♥earth]-(~/ctf/HeroCTF-v5/Web/Blogodogo-1-2)-[2023.05.13|19:59:41(HKT)]
└> unzip blog.zip      
Archive:  blog.zip
   creating: blogodogo/
  inflating: blogodogo/requirements.txt  
  inflating: blogodogo/docker-compose.yml  
  inflating: blogodogo/config.py     
   creating: blogodogo/templates/
   creating: blogodogo/templates/pages/
  inflating: blogodogo/templates/pages/add_post.html  
  inflating: blogodogo/templates/pages/register.html  
  inflating: blogodogo/templates/pages/index.html  
  inflating: blogodogo/templates/pages/404.html  
  inflating: blogodogo/templates/pages/profile.html  
  inflating: blogodogo/templates/pages/author.html  
  inflating: blogodogo/templates/pages/about.html  
  inflating: blogodogo/templates/pages/login.html  
  inflating: blogodogo/templates/pages/post.html  
   creating: blogodogo/templates/components/
  inflating: blogodogo/templates/components/flash.html  
  inflating: blogodogo/templates/components/footer.html  
  inflating: blogodogo/templates/components/base.html  
  inflating: blogodogo/templates/components/header.html  
  inflating: blogodogo/templates/components/navbar.html  
   creating: blogodogo/static/
   creating: blogodogo/static/assets/
  inflating: blogodogo/static/assets/favicon.ico  
   creating: blogodogo/static/assets/img/
  inflating: blogodogo/static/assets/img/about-bg.jpg  
  inflating: blogodogo/static/assets/img/contact-bg.jpg  
  inflating: blogodogo/static/assets/img/home-bg.jpg  
  inflating: blogodogo/static/assets/img/post-bg.jpg  
  inflating: blogodogo/static/assets/img/post-sample-image.jpg  
   creating: blogodogo/static/css/
  inflating: blogodogo/static/css/styles.css  
   creating: blogodogo/static/js/
  inflating: blogodogo/static/js/scripts.js  
   creating: blogodogo/src/
  inflating: blogodogo/src/routes.py  
  inflating: blogodogo/src/models.py  
  inflating: blogodogo/src/utils.py  
  inflating: blogodogo/src/__init__.py  
  inflating: blogodogo/src/forms.py  
   creating: blogodogo/bot/
  inflating: blogodogo/bot/bot.js    
  inflating: blogodogo/bot/package.json  
  inflating: blogodogo/Dockerfile    
  inflating: blogodogo/app.py

POST route /post/report:

@bp_routes.route("/post/report", methods=["POST"])
def report_post():
    url = request.form.get("url", "")

    if not re.match("^http://localhost:5000/.*", url):
        flash("URL not valid, please match: ^http://localhost:5000/.*", "warning")
        return redirect(url_for('bp_routes.index'))

    subprocess.run(["node", "/app/bot/bot.js", url])
    flash("Your request has been sent to an administrator!", "success")
    return redirect(url_for('bp_routes.index'))

This route will check the URL parameter is starts with http://localhost:5000/ via regular expression (regex).

/app/bot/bot.js:

// required packages
const puppeteer = require("puppeteer");

// variables
const host = process.env.HOST;
const adminUsername = process.env.ADMIN_USERNAME;
const adminPassword = process.env.ADMIN_PASSWORD;

// sleep
const delay = (time) => {
    return new Promise(resolve => setTimeout(resolve, time));
}

// navigate
async function goto(url) {
	const browser = await puppeteer.launch({
		headless: true,
		ignoreHTTPSErrors: true,
		args: [ "--no-sandbox", "--ignore-certificate-errors" ],
		executablePath: "/usr/bin/chromium-browser"
	});

	const page = await browser.newPage();
	await page.setDefaultNavigationTimeout(5000);

    // Setup bot context
    await page.goto(host + "/login");
	const username = await page.waitForSelector("#username");
	const password = await page.waitForSelector("#password");
	await username.type(adminUsername);
	await password.type(adminPassword);
	await page.keyboard.press("Enter");
    await page.waitForNavigation();

    // Go to provided URL
	try {
	    await page.goto(url);
	} catch {}

    await delay(1000);

    browser.close();
	return;
}

if (process.argv.length === 2) {
    console.error("No URL provided!");
    process.exit(1);
}

goto(process.argv[2]);

If the regex pattern is matched, it'll let the bot login as admin, and go to the provided URL.

Then, in GET route /post/<string:slug>:

@bp_routes.route("/post/<string:slug>", methods=["GET"])
def view_post(slug):
    post = Posts.query.filter(Posts.slug == slug).first()

    if not post:
        flash("This post does not exists.", "warning")
        return redirect(url_for('bp_routes.index')) 

    if post.draft and (not current_user.is_authenticated or post.author_id != current_user.id):
        flash("You cannot see draft of other users.", "warning")
        return redirect(url_for('bp_routes.index')) 

    author = Authors.query.filter_by(id=post.author_id).first()
    return render_template("pages/post.html", title="View a post", post=post, author=author)

If the post is in draft, and we're authenticated or the draft's author ID is equal to the current user's ID, we can view the draft's post.

That being said, we need to somehow redirect the admin bot to the secret blog post, and exfiltrate the content of that draft post.

But how?

In GET route /post/preview/<string:hash_preview>, we can get a preview of a post:

@bp_routes.route("/post/preview/<string:hash_preview>", methods=["GET"])
def preview_post(hash_preview):
    post = Posts.query.filter_by(hash_preview=hash_preview).first()

    if post:
        author = Authors.query.filter_by(id=post.author_id).first()
        return render_template("pages/post.html", title="Preview a post", post=post, author=author)

    flash("Unable to find the corresponding post.", "warning")
    return redirect(url_for('bp_routes.index'))

But how can we get the hash_preview value?

Hmm… Maybe if we can exploit Server-Side Request Forgery (SSRF) in the /post/report, we could view the secret blog post?

No clue about that…