17
votes

I'm trying to create a post build file copy step in VS2010 which handles path macros when they have embedded spaces. I've tried surrounding the copy commands in double quotes but I get error from when copy is invoked if $(SolutionDir) contains a space. the echoed command line in the error message does not show the double quotes.

copy "$(SolutionDir)$(Configuration)\*" "$(TargetDir)"

I also tried separately \" and "" but both of these cause the 2 character escape sequence to appear in the echoed command line? How does one properly escape a double quote in a build step?

7
Sorry, but why do you want to punish yourself in this way. Move your solution to a path without spaces. - Steve
Try escaping the speechmarks by using %22 instead of " - keyboardP
@Steve Mine project location doesn't have spaces but another team member unknowingly placed his workspace under "Documents/Microsoft Visual Studio\Projects". Looking for a solution to not having solution break based on its location. - JonN
Could you try with a batch file? Passing the arguments inside as %1 %2 and using quotation marks if needed inside the batch? - Steve
@keyboardP Tried %22 and still get error. Error 1 The command "copy %22C:\NGLS\Debug*%22 %22C:\NGLS\DauServer\bin\Debug\%22" exited with code 1. - JonN

7 Answers

12
votes

I was having trouble using double quotes with a pre-build event command in Visual Studio. I have seen the batch file solutions to this problem, but it seems a batch file would not solve all problems and is not elegant. I found the solution was to put a space before the closing double quote. The details are as follows.

The following command worked, but would not support spaces in the path:

subwcrev $(SolutionDir) $(SolutionDir)subwcrev_template.txt $(SolutionDir)version.h

I have little control over where other developers will place the solution, so I had to support spaces in the path. Trying to use quotes around paths to support spaces, I came up with the following command. It always fails.

subwcrev "$(SolutionDir)" "$(SolutionDir)subwcrev_template.txt" "$(SolutionDir)version.h"

Almost by accident, I found the solution, put a space between the last character of the path and the double quote.

subwcrev "$(SolutionDir) " "$(SolutionDir)subwcrev_template.txt " "$(SolutionDir)version.h "

This worked. I tested this in AVR Studio 6.1, which uses a Visual Studio Shell.

9
votes

Visual Studio project files are XML files. Some special characters, such as the double quote, have to be escaped by using named entities. I think they're similar to what's used for encoding strings to html.

MSDN has a reference on How To Use Reserved XML Characters in Project Files. In your example, all you would need to do to accomplish your copy is this in the .csproj/.vbproj file:

copy "$(SolutionDir)$(Configuration)\*" "$(TargetDir)"

That will wrap both paths in double quotes. You'll get errors when referencing paths with spaces and that's why the double quotes are required.

6
votes

You need to put a double quote within two double quotes.

Example of a copy file in a post build step: copy /Y """C:\source path with spaces\somefile.txt""" """C:\destination path with spaces\"""

6
votes

I couldn't get the other answers to work.

I finally just escaped the last "\":

"$(TargetDir)\"
5
votes

Please, oh please don't use post build events.
Instead, use the power of MSBuild's AfterBuild target:

Right click on your project and select Edit Project File. Add an AfterBuild event:

  <Target Name="AfterBuild">
    <ItemGroup>
      <FilesToCopy Include="$(SolutionDir)$(Configuration)\*" />
    </ItemGroup>
    <Copy SourceFiles="@(FilesToCopy)"
          DestinationFolder="$(TargetDir)"
          OverwriteReadOnlyFiles="true" SkipUnchangedFiles="false" Condition="'@(FilesToCopy)' != ''" />
  </Target>

Unlike the PostBuildEvent which executes by raw cmd.exe, BeforeBuild/AfterBuild targets run by managed code, which ensures more robust execution, better maintainability and traceability.

1
votes

It is possible to just separate the trailing \ from the " with a . like so:

"$(TargetDir)."

The . just means "the current directory" (as opposed to .. to mean "the directory one level up from the current directory"); I've never actually had a use for it until this problem hit me too!

0
votes

I had another situation. Running a tool inside the Solution Dir with two parameters containing paths with spaces. This was my solution:

"$(SolutionDir)Tool.exe" --solution-dir "$(SolutionDir)\" --project-dir "$(ProjectDir)\"

Put double quotes on the inital command, use double quotes on all paths AND use a backslash before the last double quote of each path.