2016-02-15 19:11:38 +00:00
|
|
|
##
|
2017-07-24 13:26:21 +00:00
|
|
|
# This module requires Metasploit: https://metasploit.com/download
|
2016-02-15 19:11:38 +00:00
|
|
|
# Current source: https://github.com/rapid7/metasploit-framework
|
|
|
|
##
|
|
|
|
|
2016-04-23 09:36:22 +00:00
|
|
|
class MetasploitModule < Msf::Auxiliary
|
2016-02-15 19:11:38 +00:00
|
|
|
include Msf::Exploit::Remote::HttpClient
|
|
|
|
include Msf::Auxiliary::Report
|
|
|
|
|
|
|
|
def initialize(info = {})
|
|
|
|
super(update_info(info,
|
2016-02-19 03:17:08 +00:00
|
|
|
'Name' => 'Search Engine Subdomains Collector',
|
2016-02-15 19:11:38 +00:00
|
|
|
'Description' => %q(
|
|
|
|
This module can be used to gather subdomains about a domain
|
2016-02-19 03:17:08 +00:00
|
|
|
from Yahoo, Bing.
|
2016-02-15 19:11:38 +00:00
|
|
|
),
|
|
|
|
'Author' => [ 'Nixawk' ],
|
|
|
|
'License' => MSF_LICENSE))
|
|
|
|
|
|
|
|
register_options(
|
|
|
|
[
|
2016-02-17 16:03:28 +00:00
|
|
|
OptString.new('TARGET', [ true, "The target to locate subdomains for, ex: rapid7.com, 8.8.8.8"]),
|
2016-02-19 03:17:08 +00:00
|
|
|
OptBool.new('IP_SEARCH', [ false, "Enable ip of subdomains to locate subdomains", true]),
|
|
|
|
OptBool.new('ENUM_BING', [ true, "Enable Bing Search Subdomains", true]),
|
|
|
|
OptBool.new('ENUM_YAHOO', [ true, "Enable Yahoo Search Subdomains", true])
|
2017-05-03 20:42:21 +00:00
|
|
|
])
|
2016-02-15 19:11:38 +00:00
|
|
|
|
|
|
|
deregister_options('RHOST', 'RPORT', 'VHOST', 'SSL', 'Proxies')
|
|
|
|
end
|
|
|
|
|
|
|
|
def rhost_yahoo
|
|
|
|
'search.yahoo.com'
|
|
|
|
end
|
|
|
|
|
|
|
|
def rport_yahoo
|
|
|
|
80
|
|
|
|
end
|
|
|
|
|
2016-02-19 03:17:08 +00:00
|
|
|
def rhost_bing
|
|
|
|
'global.bing.com'
|
|
|
|
end
|
|
|
|
|
|
|
|
def rport_bing
|
|
|
|
80
|
|
|
|
end
|
|
|
|
|
2016-04-23 14:45:47 +00:00
|
|
|
def valid_result?(target, subdomain)
|
|
|
|
data = Rex::Socket.is_ipv4?(target) ? domain2ip(subdomain) : subdomain
|
|
|
|
data && data.include?(target) ? true : false
|
2016-04-23 12:09:12 +00:00
|
|
|
end
|
|
|
|
|
|
|
|
def domain2ip(domain)
|
|
|
|
ips = []
|
2016-02-17 16:03:28 +00:00
|
|
|
begin
|
|
|
|
ips = Rex::Socket.getaddresses(domain)
|
|
|
|
rescue SocketError
|
|
|
|
end
|
2016-04-23 12:09:12 +00:00
|
|
|
ips
|
2016-02-17 16:03:28 +00:00
|
|
|
end
|
2016-02-15 19:11:38 +00:00
|
|
|
|
2016-02-19 03:17:08 +00:00
|
|
|
def uri2domain(uri)
|
|
|
|
begin
|
|
|
|
URI(uri).host
|
|
|
|
rescue URI::InvalidURIError
|
|
|
|
nil
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
def bing_search(dork)
|
|
|
|
print_status("Searching Bing for subdomains from #{dork}")
|
|
|
|
results = []
|
2016-04-23 14:58:39 +00:00
|
|
|
|
|
|
|
begin
|
2016-02-23 10:56:34 +00:00
|
|
|
searches = ['1', '51', '101', '151', '201', '251', '301', '351', '401', '451']
|
|
|
|
searches.each do |num|
|
|
|
|
resp = send_request_cgi!(
|
|
|
|
'rhost' => rhost_bing,
|
|
|
|
'rport' => rport_bing,
|
|
|
|
'vhost' => rhost_bing,
|
|
|
|
'method' => 'GET',
|
|
|
|
'uri' => '/search',
|
|
|
|
'vars_get' => {
|
|
|
|
'FROM' => 'HPCNEN',
|
|
|
|
'setmkt' => 'en-us',
|
|
|
|
'setlang' => 'en-us',
|
|
|
|
'first' => num,
|
|
|
|
'q' => dork
|
|
|
|
})
|
|
|
|
|
|
|
|
next unless resp && resp.code == 200
|
|
|
|
html = resp.get_html_document
|
|
|
|
matches = html.search('cite')
|
|
|
|
matches.each do |match|
|
|
|
|
result = uri2domain(match.text)
|
|
|
|
next unless result
|
|
|
|
result.to_s.downcase!
|
|
|
|
results << result
|
2016-04-23 14:58:39 +00:00
|
|
|
end
|
2016-02-19 03:17:08 +00:00
|
|
|
end
|
2016-04-23 14:58:39 +00:00
|
|
|
rescue ::Exception => e
|
|
|
|
print_error("#{dork} - #{e.message}")
|
2016-02-19 03:17:08 +00:00
|
|
|
end
|
2016-04-23 15:19:08 +00:00
|
|
|
results
|
2016-02-19 03:17:08 +00:00
|
|
|
end
|
|
|
|
|
2016-02-17 16:03:28 +00:00
|
|
|
def yahoo_search(dork)
|
|
|
|
print_status("Searching Yahoo for subdomains from #{dork}")
|
|
|
|
results = []
|
2016-02-19 03:17:08 +00:00
|
|
|
|
2016-04-23 14:58:39 +00:00
|
|
|
begin
|
|
|
|
searches = ["1", "101", "201", "301", "401", "501"]
|
|
|
|
searches.each do |num|
|
|
|
|
resp = send_request_cgi!(
|
|
|
|
'rhost' => rhost_yahoo,
|
|
|
|
'rport' => rport_yahoo,
|
|
|
|
'vhost' => rhost_yahoo,
|
|
|
|
'method' => 'GET',
|
|
|
|
'uri' => '/search',
|
|
|
|
'vars_get' => {
|
|
|
|
'pz' => 100,
|
|
|
|
'p' => dork,
|
|
|
|
'b' => num
|
2016-04-23 15:02:41 +00:00
|
|
|
})
|
2016-02-19 03:17:08 +00:00
|
|
|
|
2016-04-23 14:58:39 +00:00
|
|
|
next unless resp && resp.code == 200
|
|
|
|
html = resp.get_html_document
|
|
|
|
matches = html.search('span[@class=" fz-15px fw-m fc-12th wr-bw lh-15"]')
|
|
|
|
matches.each do |match|
|
|
|
|
result = match.text
|
|
|
|
result = result.split('/')[0]
|
|
|
|
result = result.split(':')[0]
|
|
|
|
next unless result
|
|
|
|
result.to_s.downcase!
|
|
|
|
results << result
|
|
|
|
end
|
2016-02-17 16:03:28 +00:00
|
|
|
end
|
2016-04-23 14:58:39 +00:00
|
|
|
rescue ::Exception => e
|
|
|
|
print_error("#{dork} - #{e.message}")
|
2016-02-19 03:17:08 +00:00
|
|
|
end
|
2016-02-17 16:03:28 +00:00
|
|
|
results
|
2016-02-19 03:17:08 +00:00
|
|
|
end
|
|
|
|
|
2016-04-23 14:45:47 +00:00
|
|
|
def search_subdomains(target)
|
2016-02-17 16:03:28 +00:00
|
|
|
domains = {}
|
2016-04-23 14:45:47 +00:00
|
|
|
ipv4 = Rex::Socket.is_ipv4?(target)
|
|
|
|
dork = ipv4 ? "ip:#{target}" : "domain:#{target}"
|
2016-02-19 03:17:08 +00:00
|
|
|
|
2016-04-23 15:02:41 +00:00
|
|
|
results = [] # merge results to reduce query times
|
2016-04-23 14:45:47 +00:00
|
|
|
results |= bing_search(dork) if datastore['ENUM_BING']
|
|
|
|
results |= yahoo_search(dork) if datastore['ENUM_YAHOO']
|
2016-02-17 16:03:28 +00:00
|
|
|
|
2016-04-23 11:47:33 +00:00
|
|
|
return domains if results.nil? || results.empty?
|
2016-02-17 16:03:28 +00:00
|
|
|
results.each do |subdomain|
|
|
|
|
next if domains.include?(subdomain)
|
2016-04-23 14:45:47 +00:00
|
|
|
next unless valid_result?(target, subdomain)
|
2016-02-17 16:03:28 +00:00
|
|
|
print_good("#{dork} subdomain: #{subdomain}")
|
2016-04-23 14:45:47 +00:00
|
|
|
if ipv4
|
|
|
|
domains[subdomain] = [target]
|
|
|
|
else
|
|
|
|
ips = domain2ip(subdomain)
|
|
|
|
next if ips.empty?
|
|
|
|
domains[subdomain] = ips
|
|
|
|
ips.each { |ip| search_subdomains(ip) } if !ips.empty? && datastore['IP_SEARCH']
|
|
|
|
end
|
2016-02-17 16:03:28 +00:00
|
|
|
end
|
2016-02-18 02:26:45 +00:00
|
|
|
return if domains.empty?
|
2016-04-23 14:45:47 +00:00
|
|
|
report_note(host: target, type: 'Subdomains', update: :unique_data, data: domains)
|
2016-02-15 19:11:38 +00:00
|
|
|
end
|
|
|
|
|
|
|
|
def run
|
2016-04-23 14:45:47 +00:00
|
|
|
search_subdomains(datastore['TARGET'])
|
2016-02-15 19:11:38 +00:00
|
|
|
end
|
|
|
|
end
|