2014-10-31 21:27:30 +00:00
|
|
|
# -*- coding: binary -*-
|
2014-11-03 19:46:38 +00:00
|
|
|
require 'net/dns'
|
|
|
|
|
2014-10-31 21:27:30 +00:00
|
|
|
module Msf
|
2014-11-03 19:46:38 +00:00
|
|
|
# This module provides methods for working with mDNS
|
|
|
|
module Auxiliary::MDNS
|
|
|
|
# Initializes an instance of an auxiliary module that uses mDNS
|
|
|
|
def initialize(info = {})
|
|
|
|
super
|
|
|
|
register_options(
|
2015-09-02 20:48:01 +00:00
|
|
|
[
|
|
|
|
OptAddressRange.new('RHOSTS', [true, 'The multicast address or CIDR range of targets to query', '224.0.0.251']),
|
|
|
|
Opt::RPORT(5353),
|
|
|
|
OptString.new('NAME', [true, 'The name to query', '_services._dns-sd._udp.local']),
|
|
|
|
OptString.new('TYPE', [true, 'The query type (name, # or TYPE#)', 'PTR']),
|
|
|
|
OptString.new('CLASS', [true, 'The query class (name, # or CLASS#)', 'IN'])
|
|
|
|
],
|
|
|
|
self.class
|
|
|
|
)
|
2014-11-03 19:46:38 +00:00
|
|
|
end
|
|
|
|
|
|
|
|
def setup
|
|
|
|
query_class_name
|
|
|
|
query_type_name
|
|
|
|
end
|
|
|
|
|
2015-08-13 18:29:56 +00:00
|
|
|
def build_probe
|
|
|
|
@probe ||= ::Net::DNS::Packet.new(query_name, query_type_num, query_class_num).data
|
|
|
|
# TODO: support QU vs QM probes
|
2016-04-21 18:15:58 +00:00
|
|
|
#+ @probe[@probe.size-2] = [0x80].pack('C')
|
|
|
|
#+ @probe
|
2015-02-25 00:06:07 +00:00
|
|
|
end
|
|
|
|
|
2014-11-03 19:46:38 +00:00
|
|
|
def query_class
|
|
|
|
if datastore['CLASS'] =~ /^\d+$/
|
|
|
|
datastore['CLASS'].to_i
|
|
|
|
else
|
|
|
|
datastore['CLASS'].upcase
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
def query_class_name
|
|
|
|
Net::DNS::RR::Classes.new(query_class).to_s
|
|
|
|
end
|
2014-10-31 21:27:30 +00:00
|
|
|
|
2014-11-03 19:46:38 +00:00
|
|
|
def query_class_num
|
|
|
|
Net::DNS::RR::Classes.new(query_class).to_i
|
|
|
|
end
|
|
|
|
|
|
|
|
def query_type
|
|
|
|
if datastore['TYPE'] =~ /^\d+$/
|
|
|
|
datastore['TYPE'].to_i
|
|
|
|
else
|
|
|
|
datastore['TYPE'].upcase
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
2015-02-25 00:06:07 +00:00
|
|
|
def query_name
|
|
|
|
datastore['NAME']
|
|
|
|
end
|
|
|
|
|
2014-11-03 19:46:38 +00:00
|
|
|
def query_type_name
|
|
|
|
Net::DNS::RR::Types.new(query_type).to_s
|
|
|
|
end
|
|
|
|
|
|
|
|
def query_type_num
|
|
|
|
Net::DNS::RR::Types.new(query_type).to_i
|
|
|
|
end
|
2015-09-02 17:31:46 +00:00
|
|
|
|
|
|
|
def describe_response(response)
|
|
|
|
decoded = Resolv::DNS::Message.decode(response)
|
2015-09-02 18:21:54 +00:00
|
|
|
answers = decoded.answer
|
2015-09-02 20:48:01 +00:00
|
|
|
|
2015-09-02 18:21:54 +00:00
|
|
|
if answers.empty? # not sure this will ever happen...
|
|
|
|
"no answers"
|
|
|
|
else
|
2015-09-02 20:43:26 +00:00
|
|
|
# there are often many answers for the same RR, so group them
|
2015-09-02 20:48:01 +00:00
|
|
|
grouped_answers = answers.group_by { |name, _, _| name }
|
2015-09-02 20:43:26 +00:00
|
|
|
# now summarize each group by noting the resource type and the notable
|
|
|
|
# part(s) of that RR
|
2015-09-02 20:48:01 +00:00
|
|
|
summarized_answers = grouped_answers.map do |name, these_answers|
|
|
|
|
summarized_group = these_answers.map do |_, _, data|
|
|
|
|
case data
|
|
|
|
when Resolv::DNS::Resource::IN::A
|
|
|
|
"A #{data.address}"
|
|
|
|
when Resolv::DNS::Resource::IN::AAAA
|
|
|
|
"AAAA #{data.address}"
|
|
|
|
when Resolv::DNS::Resource::IN::PTR
|
|
|
|
"PTR #{data.name}"
|
|
|
|
when Resolv::DNS::Resource::IN::SRV
|
|
|
|
"SRV #{data.target}"
|
|
|
|
when Resolv::DNS::Resource::IN::TXT
|
|
|
|
"TXT #{data.strings.join(',')}"
|
|
|
|
else
|
|
|
|
data.inspect
|
|
|
|
end
|
2015-09-02 20:43:26 +00:00
|
|
|
end
|
2015-09-02 23:30:03 +00:00
|
|
|
"#{name}: (#{summarized_group.join(", ")})"
|
2015-09-02 20:43:26 +00:00
|
|
|
end
|
2015-09-02 23:30:03 +00:00
|
|
|
summarized_answers.join(', ')
|
2015-09-02 18:21:54 +00:00
|
|
|
end
|
2015-09-02 17:31:46 +00:00
|
|
|
end
|
|
|
|
|
|
|
|
def request_info
|
|
|
|
"#{query_name} #{query_class}/#{query_type}"
|
|
|
|
end
|
2014-10-31 21:27:30 +00:00
|
|
|
end
|
|
|
|
end
|