Разнесенные пути, msbuild и psake

Следующий вопрос здесь.

Это правильно работает для компиляции приложения mvc3.

task Compile 
{
    $config = $script:siteConfig.config

    exec { & "C:\Windows\Microsoft.NET\Framework\v4.0.30319\msbuild.exe" $webproject_path `
    /p:Configuration=$config /p:WebProjectOutputDir="$publish_dir" `
    /p:Outdir="$out_dir" /p:CleanWebProjectOutputDir=False `
    /T:_WPPCopyWebApplication /T:ResolveReferences /verbosity:quiet /nologo }
}

Все эти переменные пути являются свойствами script. Однако, когда в эти рассчитанные пути вводятся пробелы (например, проект перемещается из C:\Projects\в C:\Users\ASDFG1\Documents\Visual Studio 2010\Projects), msbuild считает, что существует несколько файлов проекта. Это имеет смысл, но я должен что-то упускать, поэтому получить синтаксическую переменную в кавычки не должно быть так сложно.

Измененные варианты

exec { Invoke-Expression "& C:\Windows\Microsoft.NET\Framework\v4.0.30319\msbuild.exe '$webproject_path' /p:Configuration=$config /p:WebProjectOutputDir='$publish_dir' /p:Outdir='$out_dir' /p:CleanWebProjectOutputDir=False /T:_WPPCopyWebApplication /T:ResolveReferences /verbosity:quiet /nologo" }

exec { C:\Windows\Microsoft.NET\Framework\v4.0.30319\msbuild.exe "`"$webproject_path`"" `
/p:Configuration=$config /p:WebProjectOutputDir="`"$publish_dir`"" `
/p:Outdir="`"$out_dir`"" /p:CleanWebProjectOutputDir=False `
/T:_WPPCopyWebApplication /T:ResolveReferences /verbosity:quiet /nologo }

Ответ 1

Используя EchoArgs.exe для воспроизведения проблемы, мы видим, что кавычки не передаются исполняемому файлу по желанию:

PS> $publish_dir = 'C:\Users\Documents\Visual Studio 2010\Projects'
PS> ./echoargs /p:WebProjectOutputDir="$publish_dir"
Arg 0 is </p:WebProjectOutputDir=C:\Users\Documents\Visual Studio 2010\Projects>

PS> ./echoargs /p:WebProjectOutputDir="`"$publish_dir`""
Arg 0 is </p:WebProjectOutputDir=C:\Users\Documents\Visual>
Arg 1 is <Studio>
Arg 2 is <2010\Projects>


Используя опцию backslash-escaping из этого ответа, мы можем сохранить расширение переменной и прилагаемые кавычки:

PS> ./echoargs /p:WebProjectOutputDir=\`"$publish_dir\`"
Arg 0 is </p:WebProjectOutputDir="C:\Users\Documents\Visual Studio 2010\Projects">

Здесь обратные сигналы указывают PowerShell обрабатывать символы кавычек как литеральные значения, а обратная косая черта указывает вызов вызова для сохранения кавычек.


В качестве альтернативы мы могли бы придерживаться одного уровня экранирования, предварительно оценив полный аргумент, вместо того, чтобы вставлять переменную $publish_dir:

PS> $publishArg = '/p:WebProjectOutputDir=\"{0}\"' -f $publish_dir
PS> ./echoargs $publishArg
Arg 0 is </p:WebProjectOutputDir="C:\Users\Documents\Visual Studio 2010\Projects">