Windows Local Persistence
Room Link: https://tryhackme.com/room/windowslocalpersistence
Tampering With Unprivileged Accounts
Assign Group Memberships
For this part of the task, we will assume you have dumped the password hashes of the victim machine and successfully cracked the passwords for the unprivileged accounts in use.
The direct way to make an unprivileged user gain administrative privileges is to make it part of the Administrators group. We can easily achieve this with the following command.
Victim(cmd)
This will allow you to access the server by using RDP, WinRM or any other remote administration service available.
If this looks too suspicious, you can use the Backup Operators group. Users in this group won't have administrative privileges but will be allowed to read/write any file or registry key on the system, ignoring any configured DACL. This would allow us to copy the content of the SAM and SYSTEM registry hives, which we can then use to recover the password hashes for all the users, enabling us to escalate to any administrative account trivially.
To do so, we begin by adding the account to the Backup Operators group.
Victim(cmd)
Since this is an unprivileged account, it cannot RDP or WinRM back to the machine unless we add it to the Remote Desktop Users (RDP) or Remote Management Users (WinRM) groups. We'll use WinRM for this task.
Victim(cmd)
We'll assume we have already dumped the credentials on the server and have thmuser1's password. Let's connect via WinRM using its credentials. If you tried to connect right now from your attacker machine, you'd be surprised to see that even if you are on the Backups Operators group, you wouldn't be able to access all files as expected. A quick check on our assigned groups would indicate that we are a part of Backup Operators, but the group is disabled.
Kali
This is due to User Account Control (UAC). One of the features implemented by UAC, LocalAccountTokenFilterPolicy, strips any local account of its administrative privileges when logging in remotely. While you can elevate your privileges through UAC from a graphical user session (Read more on UAC here), if you are using WinRM, you are confined to a limited access token with no administrative privileges.
To be able to regain administration privileges from your user, we'll have to disable LocalAccountTokenFilterPolicy by changing the following registry key to 1.
Victim
Once all of this has been set up, we are ready to use our backdoor user. First, let's establish a WinRM connection and check that the Backup Operators group is enabled for our user:
Kali
----
We then proceed to make a backup of SAM and SYSTEM files and download them to our attacker machine.
Kali(WinRM)
With those files, we can dump the password hashes for all users using secretsdump.py
or other similar tools.
Kali
And finally, perform Pass-the-Hash to connect to the victim machine with Administrator privileges.
Kali
Special Privileges and Security Descriptors
A similar result to adding a user to the Backup Operators group can be achieved without modifying any group membership. Special groups are only special because the operating system assigns them specific privileges by default. Privileges are simply the capacity to do a task on the system itself. They include simple things like having the capabilities to shut down the server up to very privileged operations like being able to take ownership of any file on the system. A complete list of available privileges can be found here for reference.
In the case of the Backup Operators group, it has the following two privileges assigned by default:
SeBackupPrivilege: The user can read any file in the system, ignoring any DACL in place.
SeRestorePrivilege: The user can write any file in the system, ignoring any DACL in place.
We can assign such privileges to any user, independent of their group memberships. To do so, we can use the secedit
command. First, we will export the current configuration to a temporary file. We open the file and add our user to the lines in the configuration regarding the SeBackupPrivilege and SeRestorePrivilege.
Victim(cmd)
We finally convert the .inf file into a .sdb file which is then used to load the configuration back into the system.
Victim(cmd)
You should now have a user with equivalent privileges to any Backup Operator. The user still can't log into the system via WinRM, so let's do something about it. Instead of adding the user to the Remote Management Users group, we'll change the security descriptor associated with the WinRM service to allow thmuser2 to connect. Think of a security descriptor as an ACL but applied to other system facilities.
To open the configuration window for WinRM's security descriptor, you can use the following command in Powershell (you'll need to use the GUI session for this).
Victim(powershell)
This will open a window where you can add thmuser2 and assign it full privileges to connect to WinRM.
Once we have done this, our user can connect via WinRM. Since the user has the SeBackup and SeRestore privileges, we can repeat the steps to recover the password hashes from the SAM and connect back with the Administrator user.
Notice that for this user to work with the given privileges fully, you'd have to change the LocalAccountTokenFilterPolicy registry key, but we've done this already to get the previous flag.
If you check your user's group memberships, it will look like a regular user. Nothing suspicious at all!
RID Hijacking
Another method to gain administrative privileges without being an administrator is changing some registry values to make the operating system think you are the Administrator.
When a user is created, an identifier called Relative ID (RID) is assigned to them. The RID is simply a numeric identifier representing the user across the system. When a user logs on, the LSASS process gets its RID from the SAM registry hive and creates an access token associated with that RID. If we can tamper with the registry value, we can make windows assign an Administrator access token to an unprivileged user by associating the same RID to both accounts.
In any Windows system, the default Administrator account is assigned the RID = 500, and regular users usually have RID >= 1000.
To find the assigned RIDs for any user, you can use the following command.
Victim(cmd)
The RID is the last bit of the SID (1010 for thmuser3 and 500 for Administrator). The SID is an identifier that allows the operating system to identify a user across a domain, but we won't mind too much about the rest of it for this task.
Now we only have to assign the RID=500 to thmuser3. To do so, we need to access the SAM using Regedit. The SAM is restricted to the SYSTEM account only, so even the Administrator won't be able to edit it. To run Regedit as SYSTEM, we will use psexec, available in C:\tools\pstools
in your machine.
Victim(cmd)
From Regedit, we will go to HKLM\SAM\SAM\Domains\Account\Users\
where there will be a key for each user in the machine. Since we want to modify thmuser3, we need to search for a key with its RID in hex (1010 = 0x3F2). Under the corresponding key, there will be a value called F, which holds the user's effective RID at position 0x30:
Notice the RID is stored using little-endian notation, so its bytes appear reversed.
We will now replace those two bytes with the RID of Administrator in hex (500 = 0x01F4), switching around the bytes (F401):
The next time thmuser3 logs in, LSASS will associate it with the same RID as Administrator and grant them the same privileges.
Backdooring Files
Executable Files
If you find any executable laying around the desktop, the chances are high that the user might use it frequently. Suppose we find a shortcut to PuTTY lying around. If we checked the shortcut's properties, we could see that it (usually) points to C:\Program Files\PuTTY\putty.exe
. From that point, we could download the executable to our attacker's machine and modify it to run any payload we wanted.
You can easily plant a payload of your preference in any .exe file with msfvenom
. The binary will still work as usual but execute an additional payload silently by adding an extra thread in your binary. To create a backdoored putty.exe, we can use the following command.
Victim(cmd)
The resulting puttyX.exe will execute a reverse_tcp meterpreter payload without the user noticing it. While this method is good enough to establish persistence, let's look at other sneakier techniques.
If we don't want to alter the executable, we can always tamper with the shortcut file itself. Instead of pointing directly to the expected executable, we can change it to point to a script that will run a backdoor and then execute the usual program normally.
For this task, let's check the shortcut to calc on the Administrator's desktop. If we right-click it and go to properties, we'll see where it is pointing.
Target: powershell.exe -WindowStyle hidden C:\Windows\System32\backdoor.ps1
Before hijacking the shortcut's target, let's create a simple Powershell script in C:\Windows\System32
or any other sneaky location. The script will execute a reverse shell and then run calc.exe from the original location on the shortcut's properties.
backdoor.ps1
Kali
Hijacking File Associations
In addition to persisting through executables or shortcuts, we can hijack any file association to force the operating system to run a shell whenever the user opens a specific file type.
The default operating system file associations are kept inside the registry, where a key is stored for every single file type under HKLM\Software\Classes\
. Let's say we want to check which program is used to open .txt files; we can just go and check for the .txt
subkey and find which Programmatic ID (ProgID) is associated with it. A ProgID is simply an identifier to a program installed on the system. For .txt files, we will have the following ProgID.
We can then search for a subkey for the corresponding ProgID (also under HKLM\Software\Classes), in this case, txtfile, where we will find a reference to the program in charge of handling .txt files. Most ProgID entries will have a subkey under shell\open\command where the default command to be run for files with that extension is specified.
In this case, when you try to open a .txt file, the system will execute %SystemRoot%\system32\NOTEPAD.EXE %1, where %1 represents the name of the opened file. If we want to hijack this extension, we could replace the command with a script that executes a backdoor and then opens the file as usual. First, let's create a ps1 script with the following content and save it to C:\Windows\backdoor2.ps1.
backdoor2.ps1
Notice how in Powershell, we have to pass $args[0] to notepad, as it will contain the name of the file to be opened, as given through %1.
Now let's change the registry key to run our backdoor script in a hidden window.
Default
Finally, create a listener for your reverse shell and try to open any .txt file on the victim machine (create one if needed). You should receive a reverse shell with the privileges of the user opening the file.
Kali
Abusing Services
Windows services offer a great way to establish persistence since they can be configured to run in the background whenever the victim machine is started. If we can leverage any service to run something for us, we can regain control of the victim machine each time it is started.
A service is basically an executable that runs in the background. When configuring a service, you define which executable will be used and select if the service will automatically run when the machine starts or should be manually started.
There are two main ways we can abuse services to establish persistence: either create a new service or modify an existing one to execute our payload.
Creating backdoor services
We can create and start a service named "THMservice" using the following commands.
Victim
Note: There must be a space after each equal sign for the command to work.
The "net user" command will be executed when the service is started, resetting the Administrator's password to Passwd123
. Notice how the service has been set to start automatically (start= auto), so that it runs without requiring user interaction.
Resetting a user's password works well enough, but we can also create a reverse shell with msfvenom and associate it with the created service. Notice, however, that service executables are unique since they need to implement a particular protocol to be handled by the system. If you want to create an executable that is compatible with Windows services, you can use the exe-service
format in msfvenom.
Kali
You can then copy the executable to your target system, say in C:\Windows
and point the service's binPath to it.
Kali
Kali
Victim(cmd)
Victim(cmd)
Modifying existing services
While creating new services for persistence works quite well, the blue team may monitor new service creation across the network. We may want to reuse an existing service instead of creating one to avoid detection. Usually, any disabled service will be a good candidate, as it could be altered without the user noticing it.
You can get a list of available services using the following command.
Victim(cmd)
You should be able to find a stopped service called THMService3. To query the service's configuration, you can use the following command.
Victim(cmd)
There are three things we care about when using a service for persistence:
The executable (BINARY_PATH_NAME) should point to our payload.
The service START_TYPE should be automatic so that the payload runs without user interaction.
The SERVICE_START_NAME, which is the account under which the service will run, should preferably be set to LocalSystem to gain SYSTEM privileges.
Let's start by creating a new reverse shell with msfvenom.
Kali
Kali
Kali
Victim(cmd)
Victim(cmd)
Abusing Scheduled Tasks
Task Scheduler
The most common way to schedule tasks is using the built-in Windows task scheduler. The task scheduler allows for granular control of when your task will start, allowing you to configure tasks that will activate at specific hours, repeat periodically or even trigger when specific system events occur. From the command line, you can use schtasks
to interact with the task scheduler. A complete reference for the command can be found on Microsoft's website.
Let's create a task that runs a reverse shell every single minute. In a real-world scenario, you wouldn't want your payload to run so often, but we don't want to wait too long for this room:
Victim(cmd)
Note: Be sure to use THM-TaskBackdoor
as the name of your task, or you won't get the flag.
The previous command will create a "THM-TaskBackdoor" task and execute an nc64
reverse shell back to the attacker. The /sc
and /mo
options indicate that the task should be run every single minute. The /ru
option indicates that the task will run with SYSTEM privileges.
To check if our task was successfully created, we can use the following command.
Victim(cmd)
Making Our Task Invisible
Our task should be up and running by now, but if the compromised user tries to list its scheduled tasks, our backdoor will be noticeable. To further hide our scheduled task, we can make it invisible to any user in the system by deleting its Security Descriptor (SD). The security descriptor is simply an ACL that states which users have access to the scheduled task. If your user isn't allowed to query a scheduled task, you won't be able to see it anymore, as Windows only shows you the tasks that you have permission to use. Deleting the SD is equivalent to disallowing all users' access to the scheduled task, including administrators.
The security descriptors of all scheduled tasks are stored in HKLM\SOFTWARE\Microsoft\Windows NT\CurrentVersion\Schedule\TaskCache\Tree\
. You will find a registry key for every task, under which a value named "SD" contains the security descriptor. You can only erase the value if you hold SYSTEM privileges.
To hide our task, let's delete the SD value for the "THM-TaskBackdoor" task we created before. To do so, we will use psexec
(available in C:\tools
) to open Regedit with SYSTEM privileges.
Victim(cmd)
Location: Computer\HKEY_LOCAL_MACHINE\SOFTWARE\Microsoft\Windows NT\CurrentVersion\Schedule\TaskCache\Tree\THM-TaskBackdoor
We will then delete the security descriptor for our task.
If we try to query our service again, the system will tell us there is no such task.
Victim(cmd)
If we start an nc listener in our attacker's machine, we should get a shell back after a minute.
Kali
Logon Triggered Persistence
Some actions performed by a user might also be bound to executing specific payloads for persistence. Windows operating systems present several ways to link payloads with particular interactions. This task will look at ways to plant payloads that will get executed when a user logs into the system.
Startup folder
Each user has a folder under C:\Users\<your_username>\AppData\Roaming\Microsoft\Windows\Start Menu\Programs\Startup
where you can put executables to be run whenever the user logs in. An attacker can achieve persistence just by dropping a payload in there. Notice that each user will only run whatever is available in their folder.
If we want to force all users to run a payload while logging in, we can use the folder under C:\ProgramData\Microsoft\Windows\Start Menu\Programs\StartUp
in the same way.
For this task, let's generate a reverse shell payload using msfvenom.
Kali
We will then copy our payload into the victim machine. You can spawn an http.server
with Python3 and use wget on the victim machine to pull your file.
Kali
Kali #2
Victim(powershell)
We then store the payload into the C:\ProgramData\Microsoft\Windows\Start Menu\Programs\StartUp folder to get a shell back for any user logging into the machine/
Victim(powershell)
Now be sure to sign out of your session from the start menu (closing the RDP window is not enough as it leaves your session open).
And log back via RDP. You should immediately receive a connection back to your attacker's machine.
Run / RunOnceYou can also force a user to execute a program on logon via the registry. Instead of delivering your payload into a specific directory, you can use the following registry entries to specify applications to run at logon:
HKCU\Software\Microsoft\Windows\CurrentVersion\Run
HKCU\Software\Microsoft\Windows\CurrentVersion\RunOnce
HKLM\Software\Microsoft\Windows\CurrentVersion\Run
HKLM\Software\Microsoft\Windows\CurrentVersion\RunOnce
The registry entries under HKCU
will only apply to the current user, and those under HKLM
will apply to everyone. Any program specified under the Run
keys will run every time the user logs on. Programs specified under the RunOnce
keys will only be executed a single time.
For this task, let's create a new reverse shell with msfvenom.
Kali
Kali
Kali #2
Victim(powershell)
After transferring it to the victim machine, let's move it to C:\Windows:
Victim(powershell)
Let's then create a REG_EXPAND_SZ registry entry under HKLM\Software\Microsoft\Windows\CurrentVersion\Run. The entry's name can be anything you like, and the value will be the command we want to execute.
Note: While in a real-world set-up you could use any name for your registry entry, for this task you are required to use MyBackdoor to receive the flag.
Victim(cmd)
After doing this, sign out of your current session and log in again, and you should receive a shell (it will probably take around 10-20 seconds).
Winlogon
Another alternative to automatically start programs on logon is abusing Winlogon, the Windows component that loads your user profile right after authentication (amongst other things).
Winlogon uses some registry keys under HKLM\Software\Microsoft\Windows NT\CurrentVersion\Winlogon\
that could be interesting to gain persistence:
Userinit
points touserinit.exe
, which is in charge of restoring your user profile preferences.shell
points to the system's shell, which is usuallyexplorer.exe
.
Kali
Kali
Kali #2
Victim(powershell)
Victim(powershell)
We then alter either shell
or Userinit
in HKLM\Software\Microsoft\Windows NT\CurrentVersion\Winlogon\
. In this case we will use Userinit
, but the procedure with shell
is the same.
Note: While both shell
and Userinit
could be used to achieve persistence in a real-world scenario, to get the flag in this room, you will need to use Userinit
.
Victim(powershell)
Add: C:\Windows\revshell.exe
After doing this, sign out of your current session and log in again, and you should receive a shell (it will probably take around 10 seconds).
Logon scripts
One of the things userinit.exe
does while loading your user profile is to check for an environment variable called UserInitMprLogonScript
. We can use this environment variable to assign a logon script to a user that will get run when logging into the machine. The variable isn't set by default, so we can just create it and assign any script we like.
Notice that each user has its own environment variables; therefore, you will need to backdoor each separately.
Let's first create a reverse shell to use for this technique.
Kali
Kali
Kali #2
Victim(powershell)
Victim(powershell)
To create an environment variable for a user, you can go to its HKCU\Environment
in the registry. We will use the UserInitMprLogonScript
entry to point to our payload so it gets loaded when the users logs in:
NOTE: Remember to open registry without PsExec64 to have 'HKCU\Environment' for the current logged user. (PsExec64 starts registry with SYSTEM).
Victim(powershell)
Notice that this registry key has no equivalent in HKLM, making your backdoor apply to the current user only.
After doing this, sign out of your current session and log in again, and you should receive a shell (it will probably take around 10 seconds).
Persisting Through Existing Services
Using Web Shells
The usual way of achieving persistence in a web server is by uploading a web shell to the web directory. This is trivial and will grant us access with the privileges of the configured user in IIS, which by default is iis apppool\defaultapppool
. Even if this is an unprivileged user, it has the special SeImpersonatePrivilege
, providing an easy way to escalate to the Administrator using various known exploits. For more information on how to abuse this privilege, see the Windows Privesc Room.
Let's start by downloading an ASP.NET web shell. A ready to use web shell is provided here, but feel free to use any you prefer. Transfer it to the victim machine and move it into the webroot, which by default is located in the C:\inetpub\wwwroot
directory.
Victim(cmd)
Note: Depending on the way you create/transfer shell.aspx
, the permissions in the file may not allow the web server to access it. If you are getting a Permission Denied error while accessing the shell's URL, just grant everyone full permissions on the file to get it working. You can do so with icacls shell.aspx /grant Everyone:F
.
We can then run commands from the web server by pointing to the following URL:
http://MACHINE_IP/shell.aspx
While web shells provide a simple way to leave a backdoor on a system, it is usual for blue teams to check file integrity in the web directories. Any change to a file in there will probably trigger an alert.
Backdooring the Login Screen / RDP
If we have physical access to the machine (or RDP in our case), you can backdoor the login screen to access a terminal without having valid credentials for a machine.
We will look at two methods that rely on accessibility features to this end.
Sticky Keys
When pressing key combinations like CTRL + ALT + DEL
, you can configure Windows to use sticky keys, which allows you to press the buttons of a combination sequentially instead of at the same time. In that sense, if sticky keys are active, you could press and release CTRL
, press and release ALT
and finally, press and release DEL
to achieve the same effect as pressing the CTRL + ALT + DEL
combination.
To establish persistence using Sticky Keys, we will abuse a shortcut enabled by default in any Windows installation that allows us to activate Sticky Keys by pressing SHIFT
5 times. After inputting the shortcut, we should usually be presented with a screen that looks as follows.
After pressing SHIFT
5 times, Windows will execute the binary in C:\Windows\System32\sethc.exe
. If we are able to replace such binary for a payload of our preference, we can then trigger it with the shortcut. Interestingly, we can even do this from the login screen before inputting any credentials.
A straightforward way to backdoor the login screen consists of replacing sethc.exe
with a copy of cmd.exe
. That way, we can spawn a console using the sticky keys shortcut, even from the logging screen.
To overwrite sethc.exe
, we first need to take ownership of the file and grant our current user permission to modify it. Only then will we be able to replace it with a copy of cmd.exe
. We can do so with the following commands.
Victim(cmd)
After doing so, lock your session from the start menu.
You should now be able to press SHIFT
five times to access a terminal with SYSTEM privileges directly from the login screen.
Utilman
Utilman is a built-in Windows application used to provide Ease of Access options during the lock screen.
When we click the ease of access button on the login screen, it executes C:\Windows\System32\Utilman.exe
with SYSTEM privileges. If we replace it with a copy of cmd.exe
, we can bypass the login screen again.
To replace utilman.exe
, we do a similar process to what we did with sethc.exe
:
Victim(cmd)
To trigger our terminal, we will lock our screen from the start button.
And finally, proceed to click on the "Ease of Access" button. Since we replaced utilman.exe
with a cmd.exe
copy, we will get a command prompt with SYSTEM privileges.
Persisting Through Existing Services
If you don't want to use Windows features to hide a backdoor, you can always profit from any existing service that can be used to run code for you. This task will look at how to plant backdoors in a typical web server setup. Still, any other application where you have some degree of control on what gets executed should be backdoorable similarly. The possibilities are endless!
Using Web Shells
The usual way of achieving persistence in a web server is by uploading a web shell to the web directory. This is trivial and will grant us access with the privileges of the configured user in IIS, which by default is iis apppool\defaultapppool
. Even if this is an unprivileged user, it has the special SeImpersonatePrivilege
, providing an easy way to escalate to the Administrator using various known exploits. For more information on how to abuse this privilege, see the Windows Privesc Room.
Let's start by downloading an ASP.NET web shell. A ready to use web shell is provided here, but feel free to use any you prefer. Transfer it to the victim machine and move it into the webroot, which by default is located in the C:\inetpub\wwwroot
directory.
Kali
Victim(powershell)
Note: Depending on the way you create/transfer shell.aspx
, the permissions in the file may not allow the web server to access it. If you are getting a Permission Denied error while accessing the shell's URL, just grant everyone full permissions on the file to get it working. You can do so with icacls shell.aspx /grant Everyone:F
.
We can then run commands from the web server by pointing to the following URL:
http://$VICTIM/shell.aspx
While web shells provide a simple way to leave a backdoor on a system, it is usual for blue teams to check file integrity in the web directories. Any change to a file in there will probably trigger an alert.
Using MSSQL as a Backdoor
There are several ways to plant backdoors in MSSQL Server installations. For now, we will look at one of them that abuses triggers. Simply put, triggers in MSSQL allow you to bind actions to be performed when specific events occur in the database. Those events can range from a user logging in up to data being inserted, updated or deleted from a given table. For this task, we will create a trigger for any INSERT into the HRDB
database.
Before creating the trigger, we must first reconfigure a few things on the database. First, we need to enable the xp_cmdshell
stored procedure. xp_cmdshell
is a stored procedure that is provided by default in any MSSQL installation and allows you to run commands directly in the system's console but comes disabled by default.
To enable it, let's open Microsoft SQL Server Management Studio 18
, available from the start menu. When asked for authentication, just use Windows Authentication (the default value), and you will be logged on with the credentials of your current Windows User. By default, the local Administrator account will have access to all DBs.
Once logged in, click on the New Query button to open the query editor:
Run the following SQL sentences to enable the "Advanced Options" in the MSSQL configuration, and proceed to enable xp_cmdshell
.
After this, we must ensure that any website accessing the database can run xp_cmdshell
. By default, only database users with the sysadmin
role will be able to do so. Since it is expected that web applications use a restricted database user, we can grant privileges to all users to impersonate the sa
user, which is the default database administrator.
After all of this, we finally configure a trigger. We start by changing to the HRDB
database.
Our trigger will leverage xp_cmdshell
to execute Powershell to download and run a .ps1
file from a web server controlled by the attacker. The trigger will be configured to execute whenever an INSERT
is made into the Employees
table of the HRDB
database.
Now that the backdoor is set up, let's create evilscript.ps1
in our attacker's machine, which will contain a Powershell reverse shell.
We will need to open two terminals to handle the connections involved in this exploit:
The trigger will perform the first connection to download and execute
evilscript.ps1
. Our trigger is using port 8000 for that.The second connection will be a reverse shell on port 4454 back to our attacker machine.
Kali #1
Kali #2
With all that ready, let's navigate to http://$VICTIM/
and insert an employee into the web application. Since the web application will send an INSERT statement to the database, our TRIGGER will provide us access to the system's console.
Last updated