siunam's Website

My personal website

Home Writeups Research Blog Projects About

Microservices

Table of Contents

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

Overview

Background

I just learnt about microservices. That means my internal server is safe now right?

I'm still making my website but you can have a free preview

Alternative links: http://34.124.157.94:5014 http://34.124.157.94:5024

http://34.124.157.94:5004

Enumeration

Home page:

When we go to /, it renders "You are not an admin", "This website is under construction, only admins allowed.".

Seems empty here.

In this challenge, we can download a file:

┌[siunam♥earth]-(~/ctf/Grey-Cat-The-Flag-2023-Qualifiers/Web/Microservices)-[2023.05.19|23:15:25(HKT)]
└> file dist.zip 
dist.zip: Zip archive data, at least v1.0 to extract, compression method=store
┌[siunam♥earth]-(~/ctf/Grey-Cat-The-Flag-2023-Qualifiers/Web/Microservices)-[2023.05.19|23:15:25(HKT)]
└> unzip dist.zip 
Archive:  dist.zip
   creating: admin_page/
  inflating: admin_page/app.py       
  inflating: admin_page/dockerfile   
 extracting: admin_page/requirements.txt  
   creating: gateway/
  inflating: gateway/app.py          
  inflating: gateway/constant.py     
  inflating: gateway/dockerfile      
 extracting: gateway/requirements.txt  
   creating: homepage/
  inflating: homepage/app.py         
  inflating: homepage/dockerfile     
 extracting: homepage/requirements.txt  
   creating: homepage/templates/
  inflating: homepage/templates/base.html  
  inflating: homepage/templates/flag.html  
  inflating: homepage/templates/index.html  
  inflating: docker-compose.yml      

In docker-compose.yml, there are 3 services:

version: '3.7'

x-common-variables: &common-variables
   ADMIN_COOKIE: fake_cookie
   FLAG: grey{fake_flag}


services:
  admin:
    build: ./admin_page
    container_name: admin_page
    environment:
       <<: *common-variables
    networks:
      - backend

  homepage:
    build: ./homepage
    container_name: home_page
    environment:
       <<: *common-variables
    networks:
      - backend
      

  gateway:
    build: ./gateway
    container_name: gateway
    ports:
      - 5004:80
    networks:
      - backend

networks:
  backend: {}

Service admin and homepage have the ADMIN_COOKIE and FLAG environment variable.

First, let's look at the homepage service.

In /homepage/app.py, we can see there's only 1 / route:

from flask import Flask, request, render_template, Response, make_response
from dotenv import load_dotenv
import os

load_dotenv()
admin_cookie = os.environ.get("ADMIN_COOKIE", "FAKE_COOKIE")
FLAG = os.environ.get("FLAG", "greyctf{This_is_fake_flag}")

app = Flask(__name__)


@app.route("/")
def homepage() -> Response:
    """The homepage for the app"""
    cookie = request.cookies.get("cookie", "Guest Pleb")

    # If admin, give flag
    if cookie == admin_cookie:
        return render_template("flag.html", flag=FLAG, user="admin")

    # Otherwise, render normal page
    response = make_response(render_template("index.html", user=cookie))
    response.set_cookie("cookie", cookie)
    return response


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

In here, it checks our cookie is matched to the admin one:

    cookie = request.cookies.get("cookie", "Guest Pleb")

    # If admin, give flag
    if cookie == admin_cookie:
        return render_template("flag.html", flag=FLAG, user="admin")

Uhh… Nothing weird in here, as we can't get admin's cookie.

In admin service, it has only 1 route /:

from fastapi import FastAPI, Request, Response
from dotenv import load_dotenv
from requests import get
import os

load_dotenv()
admin_cookie = os.environ.get("ADMIN_COOKIE", "FAKE_COOKIE")

app = FastAPI()


@app.get("/")
async def index(request: Request):
    """
    The base service for admin site
    """

    # Currently Work in Progress
    requested_service = request.query_params.get("service", None)
    if requested_service is None:
        return {"message": "requested service is not found"}

    # Filter external parties who are not local
    if requested_service == "admin_page":
        return {"message": "admin page is currently not a requested service"}

    # Legit admin on localhost
    requested_url = request.query_params.get("url", None)
    if requested_url is None:
        return {"message": "URL is not found"}

    # Testing the URL with admin
    response = get(requested_url, cookies={"cookie": admin_cookie})
    return Response(response.content, response.status_code)

However, this time it's using FastAPI, not Flask.

FastAPI is a modern web framework for building RESTful APIs in Python.

In route /, the asynchronous function checks the GET parameter service is provided and not equal to admin_page.

If the pass is checked, it'll send a GET request to GET parameter url's value, and set cookie cookie's value to admin_cookie.

In gateway service, it also has 1 route:

from flask import Flask, request, abort, Response
from requests import get
from constant import routes, excluded_headers
import sys

app = Flask(__name__)


@app.route("/", methods=["GET"])
def route_traffic() -> Response:
    """Route the traffic to upstream"""
    microservice = request.args.get("service", "home_page")

    route = routes.get(microservice, None)
    if route is None:
        return abort(404)

    # Fetch the required page with arguments appended
    raw_query_param = request.query_string.decode()
    print(f"Requesting {route} with q_str {raw_query_param}", file=sys.stderr)
    res = get(f"{route}/?{raw_query_param}")

    headers = [
        (k, v) for k, v in res.raw.headers.items() if k.lower() not in excluded_headers
    ]
    return Response(res.content, res.status_code, headers)


@app.errorhandler(400)
def not_found(e) -> Response:
    """404 error"""
    return Response(f"""Error 404: This page is not found: {e}""", 404)

/gateway/constant.py:

routes = {"admin_page": "http://admin_page", "home_page": "http://home_page"}
excluded_headers = [
    "content-encoding",
    "content-length",
    "transfer-encoding",
    "connection",
]

In route /, when the GET parameter service's value is in routes from constant.py, it'll send a GET request to the provided service with arbitrary parameter.

Then, it'll check any headers are in the excluded_headers. Finally response us the service's content.

Hmm… In the excluded_headers, we see:

content-encoding
content-length
transfer-encoding
connection

The content-length and transfer-encoding is trying to prevent us doing HTTP Request Smuggling.

So… What's the goal of this challenge?

Our goal should be using the admin service to send a GET request to a URL that captures the admin cookie.

In /admin_page/app.py route, we see this:

# Filter external parties who are not local
if requested_service == "admin_page":
    return {"message": "admin page is currently not a requested service"}

If we send a GET request like /?service=admin_page, it won't let us in.

Exploitation

After fumbling around, I recalled a YouTube video from PwnFunction, it's called: "HTTP Parameter Pollution Explained".

Then, after some testing, I found that we can pollute the service GET parameter like /?service=admin_page&service=home_page, so that we can bypass this filter:

# Filter external parties who are not local
if requested_service == "admin_page":
    return {"message": "admin page is currently not a requested service"}

Nice! We now can directly access to the admin page!

Next, we need to provide the url GET parameter.

To get the flag, we can add GET parameter url=http://home_page/.

So, our full GET parameter is:

/?service=admin_page&service=home_page&url=http://home_page/

Conclusion

What we've learned:

  1. Exploiting HTTP Parameter Pollution In Flask & FastAPI To Bypass Validation