Pass complex arguments to powershell script through encoded command - powershell

I want to run nested powershell script block, with complex type arguments. I want to pass parameters to powershell scriptblock trough encoded command and encoded arguments.
I`m trying this script.
$text = "This is a test message."
$Cred = get-credential 'alex'
$Arguments = #{
Msg = $Text
Proc = $PID
Cred = $Cred
}
$Serialized = [System.Management.Automation.PSSerializer]::Serialize($Arguments)
$Bytes = [System.Text.Encoding]::Unicode.GetBytes($Serialized)
$EncodedArguments = [Convert]::ToBase64String($Bytes)
$ScriptBlock = {
param([String]$Base64)
$Serialized = [System.Text.Encoding]::Unicode.GetString([System.Convert]::FromBase64String($Base64))
$Arguments = [System.Management.Automation.PSSerializer]::Deserialize($Serialized)
Write-Host " $($Arguments.msg) FROM PID: $($Arguments.proc), cred: $( $Arguments.Cred.Username)"
}
$Bytes = [System.Text.Encoding]::Unicode.GetBytes( $ScriptBlock.ToString() )
$EncodedCommand = [Convert]::ToBase64String( $Bytes )
Start-Process -FilePath powershell -ArgumentList '-noprofile', '-noexit', `
'-EncodedCommand', $EncodedCommand, '-EncodedArguments', $EncodedArguments
Powershell process flashing, then closing.
Help me to correct this script.
PS. Why stack overflow editor replace the header of my message 'Hello team!' ? Its all right on preview.

A -EncodedArguments parameter (or something alike) doesn't exist, therefore I would simply embed your arguments as a default param value in your (encoded) scriptblock:
$Arguments = #{
Msg = "This is a test message."
Proc = $PID
Cred = (Get-Credential 'Alex')
}
$Serialized = [System.Management.Automation.PSSerializer]::Serialize($Arguments)
$Bytes = [System.Text.Encoding]::Unicode.GetBytes($Serialized)
$EncodedArguments = [Convert]::ToBase64String($Bytes)
$ScriptBlock = "param([String]`$EncodedArguments = '$EncodedArguments')" + {
$Serialized = [System.Text.Encoding]::Unicode.GetString([System.Convert]::FromBase64String($EncodedArguments))
$Arguments = [System.Management.Automation.PSSerializer]::Deserialize($Serialized)
Write-Host " $($Arguments.msg) FROM PID: $($Arguments.proc), cred: $( $Arguments.Cred.Username)"
}
$Bytes = [System.Text.Encoding]::Unicode.GetBytes( $ScriptBlock.ToString() )
$EncodedCommand = [Convert]::ToBase64String( $Bytes )
Start-Process -FilePath powershell -ArgumentList '-noprofile', '-noexit', '-EncodedCommand', $EncodedCommand

Related

Converting from Start-process to System.Diagnostics.Process

We have working powershell script ran through start-process
$EncodedCommand = '$ E r r o r . c l e a r ( )  
                         [ s t r i n g ]   $ E n c o d e d A r g u m e n t s   =   ' P A B P A G I A a g B z A C A A V g B l A H I A c w B p A G 8 A b g A 9 A C I A M Q A u A D E A L g A w A C 4 A M Q A i A C A A e A B t A G w A b g B z A D 0 A I g B o A H Q A d A B w A D o A L w A v A H M A Y w B o A G U A b Q B h A H M A L g B t A G k A Y w B y A G 8 A c w B v A G Y A d A A u A G M A b w B t A C 8 A c A B v A H c A Z Q B y A H M A a A B l A G w A b A A v A D I A M A A w A D Q A L w A w A D Q A I g A + A A 0 A C g A g A C A A P A B P A G I A a g A g A F I A Z Q B m A E k A Z A A 9 A C I A M A A i A D 4 A D Q A K A C A A I A A g A C A A P A B U A E 4 A I A B S A G U A Z g B J A G Q A P Q A i A D A A I g A + A A 0 A C g A g A C A A I A A g A C A A I A A 8 A F Q A P g B T A H k A c w B 0 A G U A b Q A u A E 8 A Y g B q A G U A Y w B 0 A F s A X Q A 8 A C 8 A V A A + A A 0 A C g A g A C A A I A A g A C A A I A A 8 A F Q A P g B T A H k A c w B 0 A G U A b Q A u A E E A c g B y A G E A e Q A 8 A C 8 A V A A + A A 0 A C g A g A C A A I A A g A C A A I A A 8 A F Q A P g B T A H k A c w B 0 A G U A b Q A u A E 8 A Y g B q A G U A Y w B 0 A D w A L w B U A D 4 A D Q A K A C A A I A A g A C A A P A A v A F Q A T g A + A A 0 A C g A g A C A A I A A g A D w A T A B T A F Q A P g A N A A o A I A A g A C A A I A A g A C A A P A B T A D 4 A d A B l A H M A d A A t A H Q A Z Q B z A H Q A L Q B 0 A G U A c w B 0 A D w A L w B T A D 4 A D Q A K A C A A I A A g A C A A I A A g A D w A T w B i A G o A I A B S A G U A Z g B J A G Q A P Q A i A D E A I g A + A A 0 A C g A g A C A A I A A g A C A A I A A g A C A A P A B E A F Q A P g A y A D A A M g A y A C 0 A M A A 3 A C 0 A M A A 3 A F Q A M A A x A D o A M A A 4 A D o A N Q A 2 A C 4 A N Q A 3 A D Q A O Q A y A D k A N Q A r A D A A M w A 6 A D A A M A A 8 A C 8 A R A B U A D 4 A D Q A K A C A A I A A g A C A A I A A g A C A A I A A 8 A E 0 A U w A + A A 0 A C g A g A C A A I A A g A C A A I A A g A C A A I A A g A D w A T w B i A G o A I A B O A D 0 A I g B E A G k A c w B w A G w A Y Q B 5 A E g A a Q B u A H Q A I g A g A F I A Z Q B m A E k A Z A A 9 A C I A M g A i A D 4 A D Q A K A C A A I A A g A C A A I A A g A C A A I A A g A C A A I A A g A D w A V A B O A C A A U g B l A G Y A S Q B k A D 0 A I g A x A C I A P g A N A A o A I A A g A C A A I A A g A C A A I A A g A C A A I A A g A C A A I A A g A D w A V A A + A E 0 A a Q B j A H I A b w B z A G 8 A Z g B 0 A C 4 A U A B v A H c A Z Q B y A F M A a A B l A G w A b A A u A E M A b w B t A G 0 A Y Q B u A G Q A c w A u A E Q A a Q B z A H A A b A B h A H k A S A B p A G 4 A d A B U A H k A c A B l A D w A L w B U A D 4 A D Q A K A C A A I A A g A C A A I A A g A C A A I A A g A C A A I A A g A C A A I A A 8 A F Q A P g B T A H k A c w B 0 A G U A b Q A u A E U A b g B 1 A G 0 A P A A v A F Q A P g A N A A o A I A A g A C A A I A A g A C A A I A A g A C A A I A A g A C A A I A A g A D w A V A A + A F M A e Q B z A H Q A Z Q B t A C 4 A V g B h A G w A d Q B l A F Q A e Q B w A G U A P A A v A F Q A P g A N A A o A I A A g A C A A I A A g A C A A I A A g A C A A I A A g A C A A I A A g A D w A V A A + A F M A e Q B z A H Q A Z Q B t A C 4 A T w B i A G o A Z Q B j A H Q A P A A v A F Q A P g A N A A o A I A A g A C A A I A A g A C A A I A A g A C A A I A A g A C A A P A A v A F Q A T g A + A A 0 A C g A g A C A A I A A g A C A A I A A g A C A A I A A g A C A A I A A 8 A F Q A b w B T A H Q A c g B p A G 4 A Z w A + A E Q A Y Q B 0 A G U A V A B p A G 0 A Z Q A 8 A C 8 A V A B v A F M A d A B y A G k A b g B n A D 4 A D Q A K A C A A I A A g A C A A I A A g A C A A I A A g A C A A I A A g A D w A S Q A z A D I A P g A y A D w A L w B J A D M A M g A + A A 0 A C g A g A C A A I A A g A C A A I A A g A C A A I A A g A D w A L w B P A G I A a g A + A A 0 A C g A g A C A A I A A g A C A A I A A g A C A A P A A v A E 0 A U w A + A A 0 A C g A g A C A A I A A g A C A A I A A 8 A C 8 A T w B i A G o A P g A N A A o A I A A g A C A A I A A g A C A A P A B C A D 4 A d A B y A H U A Z Q A 8 A C 8 A Q g A + A A 0 A C g A g A C A A I A A g A D w A L w B M A F M A V A A + A A 0 A C g A g A C A A P A A v A E 8 A Y g B q A D 4 A D Q A K A D w A L w B P A G I A a g B z A D 4 A '  
                         $ A r g u m e n t s     =   [ S y s t e m . M a n a g e m e n t . A u t o m a t i o n . P S S e r i a l i z e r ] : : D e s e r i a l i z e (   [ S y s t e m . T e x t . E n c o d i n g ] : : U n i c o d e . G e t S t r i n g ( [ S y s t e m . C o n v e r t ] : : F r o m B a s e 6 4 S t r i n g (   $ E n c o d e d A r g u m e n t s   ) )   )  
                          
                         $ S c r i p t B l o c k   =   {  
                                 # s b 1    
                                         p a r a m (   $ P a r a m 1 ,   $ P a r a m 2 ,   $ P a r a m 3   )  
                                          
                                         # r e g i o n   O u t e r   s c r i p t   a d d i t i o n  
                                                 S t a r t - T r a n s c r i p t   - p a t h   " C : \ U s e r s \ P u b l i c \ T M P \ T r a n s c r i p t . t x t "   - f o r c e 
                                                 $ G l o b a l : P S S c r i p t N a m e   =     " S c r i p t b l o c k " 
                                                 $ P S C o m m a n d P a t h   =   " C : \ U s e r s \ A l e x \ A p p D a t a \ L o c a l \ T e m p \ S c r i p t b l o c k " 
                                         # e n d r e g i o n  
  
                                         g e t - d a t e  
                                         $ U s e r   =   w h o a m i  
                                         $ R u n a s   =   T e s t - E l e v a t e d R i g h t s  
                                         w r i t e - h o s t   " U s e r   =   $ U s e r "  
                                         w r i t e - h o s t   " R u n a s   =   $ R u n a s "  
                                         w r i t e - h o s t   " P a r a m 1   =   $ P a r a m 1 "  
                                         w r i t e - h o s t   " P a r a m 2   =   $ P a r a m 2 "  
                                         w r i t e - h o s t   " P a r a m 3   =   $ P a r a m 3 "  
  
                                         # r e g i o n   O u t e r   s c r i p t   a d d i t i o n   ( r e t u r n   r e p l a c e )  
                                                 $ U s e r   |   E x p o r t - C l i X M L   - P a t h   " C : \ U s e r s \ P u b l i c \ T M P \ R e s u l t . x m l "   - F o r c e  
                                                 t r y {   s t o p - t r a n s c r i p t   } c a t c h { }  
                                                 # e x i t  
                                         # e n d r e g i o n  
                         }  
                          
                         I n v o k e - C o m m a n d   - S c r i p t B l o c k   $ S c r i p t B l o c k   - A r g u m e n t L i s t     $ A r g u m e n t s '
$Arguments = '-Noexit', '-NoLogo', '–NoProfile', '-ExecutionPolicy RemoteSigned', "-EncodedCommand $EncodedCommand"
# line below executed successfully
start-process -filepath powershell -ArgumentList $Arguments
Now we should transform our code to use System.Diagnostics.Process
$Process = New-Object System.Diagnostics.Process
$ProcessStartInfoParam = [ordered]#{
Arguments = ( "'" + ( $Arguments -join "', '" ) + "'" )
CreateNoWindow = $False
FileName = 'powershell'
WindowStyle = 'Normal'
LoadUserProfile = $False
UseShellExecute = $False
}
$ProcessStartInfo = New-Object -TypeName 'System.Diagnostics.ProcessStartInfo' -Property $ProcessStartInfoParam
$Process.StartInfo = $ProcessStartInfo
$StartResult = $Process.Start()
We have no script output and $StartResult is true. Also trying to remove '-Noexit' argument.
I want to have new window with output like in first example.
You are joining the arguments incorrectly. Command-line arguments must be separated by space and quoted using double-quotation marks. In your case you don't need quoting as the arguments don't contain whitespace.
Try this:
$ProcessStartInfoParam = [ordered]#{
Arguments = $Arguments -join ' '
CreateNoWindow = $False
FileName = 'powershell'
WindowStyle = 'Normal'
LoadUserProfile = $False
UseShellExecute = $False
}

Elevate a script with parameters

I have a Powershell script with parameters that I'd like to be able to self-elevate.
[CmdletBinding()]
param (
[Parameter(ParameterSetName="cp")]
[Switch]
$copy = $false,
[Parameter(ParameterSetName="mv")]
[Switch]
$move = $false
)
# Elevate if required
if (-not ([Security.Principal.WindowsPrincipal] [Security.Principal.WindowsIdentity]::GetCurrent()).IsInRole([Security.Principal.WindowsBuiltInRole] 'Administrator')) {
if ([int](Get-CimInstance -Class Win32_OperatingSystem | Select-Object -ExpandProperty BuildNumber) -ge 6000) {
$Cmd = (
'-File',
"`"$($MyInvocation.MyCommand.Path)`"",
$MyInvocation.BoundParameters
)
$ProcArgs = #{
FilePath = 'PowerShell.exe'
Verb = 'RunAs'
ArgumentList = $Cmd
}
Start-Process #ProcArgs
Exit
}
}
Set-Location -LiteralPath $PSScriptRoot
Write-Host "$copy"
Pause
If I comment out the param block and run script.ps1 -copy, an elevated Powershell window opens and prints out Press enter to continue, i.e. it works.
If I comment out the if statement, the current window outputs True, i.e. it also works.
If I run the whole thing though, the elevated windows opens for a split second, then closes ignoring Pause with no output anywhere.
I want the elevated window to open and print out True.
I tested this on Linux and worked for me but couldn't test on Windows, I don't see other way around having to manipulate the $PSBoundParameters into strings to pass the arguments on -ArgumentList.
Below code is meant to be exclusively for testing, hence why I've removed the if conditions.
[CmdletBinding()]
param (
[Parameter(ParameterSetName="cp")]
[Switch]$copy,
[Parameter(ParameterSetName="mv")]
[Switch]$move
)
$argument = #(
"-File $PSCommandPath"
"-$($PSBoundParameters.Keys)"
)
$ProcArgs = #{
FilePath = 'powershell.exe'
Verb = 'RunAs'
ArgumentList = $argument
}
Start-Process #ProcArgs
"Started new Process with the argument: -$($PSBoundParameters.Keys)"
[System.Console]::ReadKey()
Exit

Start-Process and powershell.exe with splatting

I've been trying for a couple of days now to multi-thread a WPF GUI which will run a PS3.0 script once the button has been clicked. I cannot use start-job as that I would have to track (multiple sessions at once), however, I would like to just run the script in a separate process of PS- as if I were to open multiple instances of the script from a shortcut. And be able to just have an open PS window which will track the progress within the script itself.
Expected results would be starting a script in powershell.exe session and passing 3 arguments - 2 strings and 1 boolean value. Which are provided by the user.
So in ISE:
C:\temp\test.ps1 -argumentlist $computername $username $citrixtest
Works fine.
I've spent a few hours scouring through the internet only to find a thread where a start-job was recommended or a way to use a background worker- this is not what I want from the script.
So I would guess the invocation from a button click would be something of the like (some of the things I have tried)
$ComputerName = "testtext1"
$UserName = "testtext2"
$CitrixTest = $True
$command = "c:\temp\test.ps1"
$arg = #{
Computername = "$computername";
Username = "$username";
CitrixTest = "$citrixtest"
}
#$WPFStartButton.Add_Click({
Start-Process powershell -ArgumentList "-noexit -command & {$command} -argumentlist $arg"
#})
Does not pass arguments to test.ps1- it is, however, getting to the "pause" - so the script successfully launches.
Where test.ps1 is
$ComputerName
$UserName
$CitrixTest
pause
Caller:
function Caller {
Param (
$ScriptPath = "c:\temp\test.ps1"
)
$Arguments = #()
$Arguments += "-computername $ComputerName"
$Arguments += "-UserName $UserName"
$Arguments += "-citrixtest $citrixtest"
$StartParams = #{
ArgumentList = "-File ""$ScriptPath""" + $Arguments
}
Start-Process powershell #StartParams
}
Caller
Does not start the script altogether- PS window just closes- possibly a path to .ps1 script not being found.
And a different approach which also nets in the script starts but not passing the arguments
$scriptFile = '"C:\temp\test.ps1"'
[string[]]$argumentList = "-file"
$argumentList += $scriptFile
$argumentlist += $computername
$argumentlist += $UserName
$argumentlist += $CitrixTest
$start_Process_info = New-Object System.Diagnostics.ProcessStartInfo
$start_Process_info.FileName = "$PSHOME\PowerShell.exe"
$start_Process_info.Arguments = $argumentList
$newProcess = New-Object System.Diagnostics.Process
$newProcess.StartInfo = $start_Process_info
$newProcess.Start() | Out-Null
Is there a way to make this work as I want it to? Or should I just dig deeper into runspaces and try with that?
#Bill_Stewart I just realized I did not put the param(args) in my script...
And that's why it would not pull those variables as I would like them to. I will have to check when I'm back in the office if it's just that what I was missing.
Checked on my laptop that's running PS 5.1 and this seems to be working as intended
$testarg = #(
'-File'
"C:\temp\test.ps1"
"$computername"
"$username"
"$citrixtest"
)
Start-Process powershell.exe -ArgumentList $testarg
Where test.ps1 is:
param(
$ComputerName,
$UserName,
$citrixtest
)
$ComputerName
$UserName
$CitrixTest
pause

installing msi file using powershell bringing up command line options

I am trying to run the below command and it opens windows installers parameters, what am I doing wrong?
$current_pc = "$env:computername"
$filePath = "c:\users\ArA\Desktop\parameters.csv"
$machineParams = Import-CSV $filePath
$info = if($machineParams){$machineParams | where {$_.branch -eq $($current_pc.Substring($current_pc.length - 3,3))}}
else{write-output "CSV not loaded"}
$info
start-process msiexec.exe -Wait -ArgumentList /L* "c:\users\Ara\Desktop\t.log" /qn /I "c:\users\Ar\Desktop\setup.msi" DATABASE_ID=$($info.DATABASE_ID),ODBC_DATABASE_NAME=$($info.ODBC_DATABASE_NAME),ODBC_ENGINE_NAME=$($info.ODBC_ENGINE_NAME) ODBC_HOST_NAME=$($info.ODBC_HOST_NAME) NOLAN=1
It's likely how you are passing your arguments. As a best-practice, I utilize an array when passing exe arguments so they're interpreted properly:
$machineParams = Import-Csv -Path $Env:UserProfile\Desktop\parameters.csv
$info = if ($machineParams) {
$machineParams |
Where-Object { $_.branch -eq $Env:ComputerName.Substring($Env:ComputerName.Length - 3, 3) }
} else {
'CSV not loaded.'
}
$info
$StartArgs = #{
'Wait' = $true
'FilePath' = "$Env:SystemRoot\System32\msiexec.exe"
'ArgumentList' = #(
'/i',"$Env:UserProfile\Desktop\setup.msi"
'/qn'
'/log',"$Env:UserProfile\Desktop\t.log"
)
}
if ($info -ne 'CSV not loaded.') {
$StartArgs['ArgumentList'] += #(
"DATABASE_ID=$($info.DATABASE_ID)"
"ODBC_DATABASE_NAME=$($info.ODBC_DATABASE_NAME)"
"ODBC_ENGINE_NAME=$($info.ODBC_ENGINE_NAME)"
"ODBC_HOST_NAME=$($info.ODBC_HOST_NAME)"
'NOLAN=1'
)
}
Start-Process #StartArgs

Invoke powershell script from another

How can I invoke a powershell script from within another script?
This is not working:
$param1 = "C:/Users/My Folder/file1"
$param2 = "C:/Users/My Folder/file2"
$command = "C:/Users/My Folder/second.ps1"
Invoke-expression $command -File1 $param1 -File2 $param2
...
Second.ps1:
param(
[string]File1, [string]File2)...
If there are no spaces:
Invoke-expression "$command $param1 $param2"
If you know where the spaces are:
Invoke-expression "$command `$param1HasSpaces` $param2"
NB: If your execution policy is restricted (check with get-executionpolicy use:
Invoke-Expression "powershell -executionpolicy bypass -command `"$command $param1 $param2`""
You can do it like this if you slightly change your approach. Basically create the command string you want to execute, then create a scriptblock object from that and then use Invoke-Command instead of Invoke-Expression.
$param1 = "C:/Users/My Folder/file1"
$param2 = "C:/Users/My Folder/file2"
$command = "C:/Users/My Folder/second.ps1"
$str = '{0} -File1 "{1}" -File2 "{2}"' -f ($command, $param1, $param2)
$sb = [scriptblock]::Create($str)
Invoke-Command -ScriptBlock $sb