PortSwigger's "Web shell upload via Content-Type restriction bypass" Walkthrough
This is the next of PortSwigger’s file upload labs. This one is only slightly more difficult because of a Content-Type check being performed by the server.
You’ll need Burp Suite set up before following this walkthrough. You’ll also need a Portswigger Academy account. Once logged into your account, view the lab at https://portswigger.net/web-security/file-upload/lab-file-upload-web-shell-upload-via-content-type-restriction-bypass
. This is accessible from the “all labs” view or from the File Upload page.
Challenge Information
data:image/s3,"s3://crabby-images/5cbb5/5cbb56cfcab635a9041e0717535438b5b4cdbce3" alt=""
Click the “Access the Lab” button and you will be taken to a temporary website that is created for your account. This will be in format https://<random string here>.web-security-academy.net/
.
As before, we’ll to upload a PHP script as our avatar. This PHP code will execute once visited in a browser, allowing us to exfiltrate the contents of /home/carlos/secret
.
Here’s the website:
data:image/s3,"s3://crabby-images/4ebc2/4ebc2297b88f08aed4cf51035fa3fd75b12b39ef" alt=""
If we login with provided credentials wiener:peter
, we can upload avatar photos via the /my-account
view:
data:image/s3,"s3://crabby-images/394ed/394edab00d0bff8d5e4446ec88f4abe4980c5339" alt=""
Lab Solution
Open a text editor and save the following contents in a file named webshell.php
:
<?php echo file_get_contents('/home/carlos/secret'); ?>
Then try to upload the file:
data:image/s3,"s3://crabby-images/f15c7/f15c7d96e73d69327c6d0c6841204052cb4b8380" alt=""
It says that we’re restricted to two allowed Content-Types
. These are image/jpeg
and image/png
. We uploaded one that’s text/php
.
In Burp Suite, find the request in Proxy > HTTP proxy
, right-click it, and then Send to Repeater
.
data:image/s3,"s3://crabby-images/f63dc/f63dc59a45bde78a67a04ecbf019b4f1f9696a56" alt=""
The request defines the Content-Types
in the form data. We can change the value for our web shell to image/jpeg
.
data:image/s3,"s3://crabby-images/73552/73552cabcc8ef9a3750cab57a19b1383922a5b3c" alt=""
Click send, and see that it was successful.
data:image/s3,"s3://crabby-images/5b6b4/5b6b486f978d5bd0df3ecfd98dd509b49b52b179" alt=""
In your browser, reload the /my-account
page and then right-click the broken avatar image (broken because it’s not a real image file), and open in a new tab:
data:image/s3,"s3://crabby-images/e72da/e72da61022fd99fb1ec9386dba768a9573492691" alt=""
This new tab is the PHP file we uploaded, which will execute to get the file contents and then echo them as output.
data:image/s3,"s3://crabby-images/4cd21/4cd21b5b6e557d9cd9016e3bcd60ec61eeb09972" alt=""
Take this value (GCHjG786pKA2IND9uMO3M4grtctSChaB
) and submit it:
data:image/s3,"s3://crabby-images/30deb/30debbd2605c15f2cb292310c9c16cbbc599b5ae" alt=""
The lab is now solved!
data:image/s3,"s3://crabby-images/1da9c/1da9c7b26f93f8308c1a22e4579bfe16c2228dcc" alt=""