2010-11-09 06:24:32 +00:00
##
# This file is part of the Metasploit Framework and may be subject to
# redistribution and commercial restrictions. Please see the Metasploit
2012-02-21 01:40:50 +00:00
# web site for more information on licensing and terms of use.
# http://metasploit.com/
2010-11-09 06:24:32 +00:00
##
require 'msf/core'
class Metasploit3 < Msf :: Auxiliary
include Msf :: Auxiliary :: Report
2012-11-04 06:13:38 +00:00
include Msf :: Auxiliary :: UDPScanner
2010-11-09 06:24:32 +00:00
def initialize
super (
2012-11-04 06:13:38 +00:00
'Name' = > 'UPnP SSDP M-SEARCH Information Discovery' ,
'Description' = > 'Discover information from UPnP-enabled systems' ,
2013-01-29 07:05:18 +00:00
'Author' = > [ 'todb' , 'hdm' ] , # Original scanner module and vuln info reporter, respectively
2010-11-09 06:24:32 +00:00
'License' = > MSF_LICENSE
)
2012-11-04 06:13:38 +00:00
register_options ( [
Opt :: RPORT ( 1900 ) ,
OptBool . new ( 'REPORT_LOCATION' , [ true , 'This determines whether to report the UPnP endpoint service advertised by SSDP' , false ] )
] , self . class )
2010-11-09 06:24:32 +00:00
end
2013-01-29 06:55:45 +00:00
def rport
datastore [ 'RPORT' ]
end
2012-11-04 06:13:38 +00:00
def setup
super
@msearch_probe =
" M-SEARCH * HTTP/1.1 \r \n " +
" Host:239.255.255.250:1900 \r \n " +
" ST:upnp:rootdevice \r \n " +
" Man: \" ssdp:discover \" \r \n " +
" MX:3 \r \n " +
2013-02-04 03:06:57 +00:00
" \r \n "
2010-11-09 06:24:32 +00:00
end
2012-11-04 06:13:38 +00:00
def scanner_prescan ( batch )
print_status ( " Sending UPnP SSDP probes to #{ batch [ 0 ] } -> #{ batch [ - 1 ] } ( #{ batch . length } hosts) " )
@results = { }
2010-11-09 06:24:32 +00:00
end
2012-11-04 06:13:38 +00:00
def scan_host ( ip )
2013-01-29 06:55:45 +00:00
vprint_status " #{ ip } : #{ rport } - SSDP - sending M-SEARCH probe "
2012-11-04 06:13:38 +00:00
scanner_send ( @msearch_probe , ip , datastore [ 'RPORT' ] )
2010-11-09 06:24:32 +00:00
end
2012-11-08 12:08:49 +00:00
def scanner_postscan ( batch )
2013-01-29 06:55:45 +00:00
print_status " No SSDP endpoints found. " if @results . empty?
2012-11-08 12:08:49 +00:00
@results . each_pair do | skey , res |
sinfo = res [ :service ]
next unless sinfo
bits = [ ]
[ :server , :location , :usn ] . each do | k |
bits << res [ :info ] [ k ] if res [ :info ] [ k ]
end
desc = bits . join ( " | " )
sinfo [ :info ] = desc
2013-01-29 06:15:39 +00:00
res [ :vulns ] = [ ]
if res [ :info ] [ :server ] . to_s =~ / MiniUPnPd \/ 1 \ .0([ \ . \ , \ - \ ~ \ s]|$) /mi
res [ :vulns ] << {
:name = > " MiniUPnPd ProcessSSDPRequest() Out of Bounds Memory Access Denial of Service " ,
:refs = > [ 'CVE-2013-0229' ]
}
end
if res [ :info ] [ :server ] . to_s =~ / MiniUPnPd \/ 1 \ .[0-3]([ \ . \ , \ - \ ~ \ s]|$) /mi
res [ :vulns ] << {
:name = > " MiniUPnPd ExecuteSoapAction memcpy() Remote Code Execution " ,
:refs = > [ 'CVE-2013-0230' ] ,
:port = > res [ :info ] [ :ssdp_port ] || 80 ,
:proto = > 'tcp'
}
end
if res [ :info ] [ :server ] . to_s =~ / Intel SDK for UPnP devices.*|Portable SDK for UPnP devices( \/ ? \ s*$| \/ 1 \ .([0-5] \ ..*|8 \ .0.*|(6 \ .[0-9]|6 \ .1[0-7])([ \ . \ , \ - \ ~ \ s]|$))) /mi
res [ :vulns ] << {
:name = > " Portable SDK for UPnP Devices unique_service_name() Remote Code Execution " ,
:refs = > [ 'CVE-2012-5958' , 'CVE-2012-5959' ]
}
end
if res [ :vulns ] . length > 0
vrefs = [ ]
res [ :vulns ] . each do | v |
v [ :refs ] . each do | r |
vrefs << r
end
end
print_good ( " #{ skey } SSDP #{ desc } | vulns: #{ res [ :vulns ] . count } ( #{ vrefs . join ( " , " ) } ) " )
else
print_status ( " #{ skey } SSDP #{ desc } " )
end
2012-11-08 12:08:49 +00:00
report_service ( sinfo )
2013-01-29 06:15:39 +00:00
res [ :vulns ] . each do | v |
report_vuln (
:host = > sinfo [ :host ] ,
:port = > v [ :port ] || sinfo [ :port ] ,
:proto = > v [ :proto ] || 'udp' ,
:name = > v [ :name ] ,
:info = > res [ :info ] [ :server ] ,
:refs = > v [ :refs ]
)
end
2012-11-08 12:08:49 +00:00
if res [ :info ] [ :ssdp_host ]
report_service (
:host = > res [ :info ] [ :ssdp_host ] ,
:port = > res [ :info ] [ :ssdp_port ] ,
:proto = > 'tcp' ,
:name = > 'upnp' ,
:info = > res [ :info ] [ :location ] . to_s
) if datastore [ 'REPORT_LOCATION' ]
2013-01-04 08:29:34 +00:00
end
2012-11-08 12:08:49 +00:00
end
end
2012-11-04 06:13:38 +00:00
def scanner_process ( data , shost , sport )
2010-11-09 06:24:32 +00:00
2012-11-04 06:13:38 +00:00
skey = " #{ shost } : #{ datastore [ 'RPORT' ] } "
2010-11-09 06:24:32 +00:00
2012-11-08 12:08:49 +00:00
@results [ skey ] || = {
:info = > { } ,
:service = > {
:host = > shost ,
:port = > datastore [ 'RPORT' ] ,
:proto = > 'udp' ,
:name = > 'ssdp'
}
}
2013-02-06 17:27:58 +00:00
if data =~ / ^Server:[ \ s]*(.*) /i
2012-11-08 12:08:49 +00:00
@results [ skey ] [ :info ] [ :server ] = $1 . strip
2012-11-04 06:13:38 +00:00
end
2010-11-09 06:24:32 +00:00
2012-11-04 06:13:38 +00:00
ssdp_host = nil
ssdp_port = 80
location_string = ''
2013-02-06 17:27:58 +00:00
if data =~ / ^Location:[ \ s]*(.*) /i
2012-11-04 06:13:38 +00:00
location_string = $1
2012-11-08 12:08:49 +00:00
@results [ skey ] [ :info ] [ :location ] = $1 . strip
2012-11-04 06:13:38 +00:00
if location_string [ / (https?): \ x2f \ x2f([^ \ x5c \ x2f]+) / ]
ssdp_host , ssdp_port = $2 . split ( " : " ) if $2 . respond_to? ( :split )
if ssdp_port . nil?
ssdp_port = ( $1 == " http " ? 80 : 443 )
2010-11-09 06:24:32 +00:00
end
2012-11-08 12:08:49 +00:00
if ssdp_host and ssdp_port
@results [ skey ] [ :info ] [ :ssdp_host ] = ssdp_host
@results [ skey ] [ :info ] [ :ssdp_port ] = ssdp_port . to_i
end
2010-11-09 06:24:32 +00:00
end
end
2013-02-06 17:27:58 +00:00
if data =~ / ^USN:[ \ s]*(.*) /i
2012-11-08 12:08:49 +00:00
@results [ skey ] [ :info ] [ :usn ] = $1 . strip
2012-11-04 06:13:38 +00:00
end
2010-11-09 06:24:32 +00:00
end
end