I'm trying to set a public property in an installshield installer with a value containing space. While running the MSI installer I'm using below command on PowerShell command prompt. Since the value contains spaces so I used double quotes to pass the value
msiexec -i "myinstaller.msi" MYDIRPATH="C:\new folder\data.txt"
It breaks the command as the argument value C:\new folder\data.txt
has a space in the string new folder
and results in below error prompt of msiexec which is suggestive of the fact that arguments passed to the msiexec command is problematic:
If I run the very same command on windows default command shell prompt then it works like a charm.
The options that I've tried:
- Using single quote in place of double quotes
- Using a back tick (`) character before space in the argument as per this answer.
Try with this
msiexec -i "myinstaller.msi" MYDIRPATH=`"C:\new folder\data.txt`"
The escape character in PowerShell is the grave-accent(`).
To complement Marko Tica's helpful answer:
Calling external utilities in PowerShell is notoriously difficult, because PowerShell, after having done its own parsing first, rebuilds the command line that is actually invoked behind the scenes, and it's far from obvious what rules are applied.
To help with this problem, PSv3+ offers --%
, the stop-parsing symbol, which is the perfect fit here, given that the command line contains no references to PowerShell variables or subexpressions: It passes the rest of the command line as-is to the external utility, save for potential expansion of %...%
-style environment variables:
# Everything after --% is passed as-is.
msiexec --% -i "myinstaller.msi" MYDIRPATH="C:\new folder\data.txt"
As for what actually happened in your original attempt:
PowerShell translated
MYDIRPATH="C:\new folder\data.txt"
into
"MYDIRPATH=C:\new folder\data.txt"
behind the scenes - note how the entire token is now enclosed in "..."
.
Arguably, these two forms should be considered equivalent by msiexec
, but all bets are off in the anarchic world of Windows command-line argument parsing.
This is the best way to install a program in general with Powershell.
Here's an example from my own script:
start-process "c:\temp\SQLClient\sqlncli (x64).msi" -argumentlist "/qn IACCEPTSQLNCLILICENSETERMS=YES" -wait
Use Start-Process "Path\to\file\file.msi or .exe" -argumentlist (Parameters) "-qn or whatever" -wait
.
Now -wait is important, if you have a script with a slew of programs being installed, the wait command, like piping to Out-Null, will force Powershell to wait until the program finishes installing before continuing forward.