У меня есть некое сложное решение, но оно работает:
При вызове MSBuild с файлом решения он внутренне генерирует файл проекта из этого решения и создает этот файл проекта. При установке для переменной среды msbuildemitsolution значения 1 он выводит этот файл проекта на жесткий диск с именем, аналогичным файлу решения, но с последующим добавлением .proj.
Кроме того, он всегда генерирует файл .cache, который не позволяет ему постоянно создавать файл проекта.
Итак, что я делаю первым:
IF EXIST xyz.sln.proj DEL xyz.sln.proj
IF EXIST xyz.sln.cache DEL xyz.sln.cache
SET msbuildemitsolution=1
MSBuild.exe xyz-sln /t:_xxxxx_ /noconsolelogger
Где цель xxxx НЕ должна существовать. Затем MSBuild генерирует xyz.sln.proj и выводит много ошибок из-за отсутствия цели. Я игнорирую ошибки, потому что мне нужен был файл проекта.
На самом деле я делаю все вышеперечисленное в рамках программы на C #, но для облегчения понимания я выбрал здесь синтаксис BAT.
xyz.sln.proj состоит из множества целей: "Построить", "Очистить", "Восстановить" и "Опубликовать" и "", ": Очистить", ": Восстановить", ": Опубликовать", "" ...
Меня интересуют только цели "Build", "Rebuild" и "Clean".
Эти цели состоят из нескольких задач MSBuild, которые создают наборы файлов csproj. Эти наборы называются BuildLevelX (где X - число). Для проектов C ++ цели Build / Rebuild / Clean содержат задачи CallTarget, которые непосредственно вызывают задачи.
Фрагмент из моего файла:
<Target Name="Build" Condition="'$(CurrentSolutionConfigurationContents)' != ''" Outputs="@(CollectedBuildOutput)">
<MSBuild Condition="@(BuildLevel0) != ''" Projects="@(BuildLevel0)" Properties="Configuration=%(Configuration); Platform=%(Platform); BuildingSolutionFile=true; CurrentSolutionConfigurationContents=$(CurrentSolutionConfigurationContents); SolutionDir=$(SolutionDir); SolutionExt=$(SolutionExt); SolutionFileName=$(SolutionFileName); SolutionName=$(SolutionName); SolutionPath=$(SolutionPath)" ToolsVersion="$(ProjectToolsVersion)" BuildInParallel="true" UnloadProjectsOnCompletion="$(UnloadProjectsOnCompletion)" UseResultsCache="$(UseResultsCache)" ContinueOnError="true">
<Output TaskParameter="TargetOutputs" ItemName="CollectedBuildOutput" />
</MSBuild>
<Message Text="The project "%(SkipLevel0.Identity)" is not selected for building in solution configuration "$(Configuration)|$(Platform)"." Condition="@(SkipLevel0) != ''" />
<Warning Text="The project configuration for project "%(MissingConfigLevel0.Identity)" was not specified in the solution file for the solution configuration "$(Configuration)|$(Platform)"." Code="MSB4121" HelpKeyword="MSBuild.SolutionProjectConfigurationMissing" Condition="@(MissingConfigLevel0) != ''" />
<MSBuild Condition="@(BuildLevel1) != ''" Projects="@(BuildLevel1)" Properties="Configuration=%(Configuration); Platform=%(Platform); BuildingSolutionFile=true; CurrentSolutionConfigurationContents=$(CurrentSolutionConfigurationContents); SolutionDir=$(SolutionDir); SolutionExt=$(SolutionExt); SolutionFileName=$(SolutionFileName); SolutionName=$(SolutionName); SolutionPath=$(SolutionPath)" ToolsVersion="$(ProjectToolsVersion)" BuildInParallel="true" UnloadProjectsOnCompletion="$(UnloadProjectsOnCompletion)" UseResultsCache="$(UseResultsCache)" ContinueOnError="true">
<Output TaskParameter="TargetOutputs" ItemName="CollectedBuildOutput" />
</MSBuild>
<Message Text="The project "%(SkipLevel1.Identity)" is not selected for building in solution configuration "$(Configuration)|$(Platform)"." Condition="@(SkipLevel1) != ''" />
<Warning Text="The project configuration for project "%(MissingConfigLevel1.Identity)" was not specified in the solution file for the solution configuration "$(Configuration)|$(Platform)"." Code="MSB4121" HelpKeyword="MSBuild.SolutionProjectConfigurationMissing" Condition="@(MissingConfigLevel1) != ''" />
<CallTarget Targets="SomeLib" RunEachTargetSeparately="true" ContinueOnError="true" />
<MSBuild Condition="@(BuildLevel2) != ''" Projects="@(BuildLevel2)" Properties="Configuration=%(Configuration); Platform=%(Platform); BuildingSolutionFile=true; CurrentSolutionConfigurationContents=$(CurrentSolutionConfigurationContents); SolutionDir=$(SolutionDir); SolutionExt=$(SolutionExt); SolutionFileName=$(SolutionFileName); SolutionName=$(SolutionName); SolutionPath=$(SolutionPath)" ToolsVersion="$(ProjectToolsVersion)" BuildInParallel="true" UnloadProjectsOnCompletion="$(UnloadProjectsOnCompletion)" UseResultsCache="$(UseResultsCache)" ContinueOnError="true">
<Output TaskParameter="TargetOutputs" ItemName="CollectedBuildOutput" />
</MSBuild>
<Message Text="The project "%(SkipLevel2.Identity)" is not selected for building in solution configuration "$(Configuration)|$(Platform)"." Condition="@(SkipLevel2) != ''" />
<Warning Text="The project configuration for project "%(MissingConfigLevel2.Identity)" was not specified in the solution file for the solution configuration "$(Configuration)|$(Platform)"." Code="MSB4121" HelpKeyword="MSBuild.SolutionProjectConfigurationMissing" Condition="@(MissingConfigLevel2) != ''" />
<CallTarget Targets="SomeProgram" RunEachTargetSeparately="true" ContinueOnError="true" />
<MSBuild Condition="@(BuildLevel3) != ''" Projects="@(BuildLevel3)" Properties="Configuration=%(Configuration); Platform=%(Platform); BuildingSolutionFile=true; CurrentSolutionConfigurationContents=$(CurrentSolutionConfigurationContents); SolutionDir=$(SolutionDir); SolutionExt=$(SolutionExt); SolutionFileName=$(SolutionFileName); SolutionName=$(SolutionName); SolutionPath=$(SolutionPath)" ToolsVersion="$(ProjectToolsVersion)" BuildInParallel="true" UnloadProjectsOnCompletion="$(UnloadProjectsOnCompletion)" UseResultsCache="$(UseResultsCache)" ContinueOnError="true">
<Output TaskParameter="TargetOutputs" ItemName="CollectedBuildOutput" />
</MSBuild>
<Message Text="The project "%(SkipLevel3.Identity)" is not selected for building in solution configuration "$(Configuration)|$(Platform)"." Condition="@(SkipLevel3) != ''" />
<Warning Text="The project configuration for project "%(MissingConfigLevel3.Identity)" was not specified in the solution file for the solution configuration "$(Configuration)|$(Platform)"." Code="MSB4121" HelpKeyword="MSBuild.SolutionProjectConfigurationMissing" Condition="@(MissingConfigLevel3) != ''" />
<MSBuild Condition="@(BuildLevel4) != ''" Projects="@(BuildLevel4)" Properties="Configuration=%(Configuration); Platform=%(Platform); BuildingSolutionFile=true; CurrentSolutionConfigurationContents=$(CurrentSolutionConfigurationContents); SolutionDir=$(SolutionDir); SolutionExt=$(SolutionExt); SolutionFileName=$(SolutionFileName); SolutionName=$(SolutionName); SolutionPath=$(SolutionPath)" ToolsVersion="$(ProjectToolsVersion)" BuildInParallel="true" UnloadProjectsOnCompletion="$(UnloadProjectsOnCompletion)" UseResultsCache="$(UseResultsCache)" ContinueOnError="true">
<Output TaskParameter="TargetOutputs" ItemName="CollectedBuildOutput" />
</MSBuild>
<Message Text="The project "%(SkipLevel4.Identity)" is not selected for building in solution configuration "$(Configuration)|$(Platform)"." Condition="@(SkipLevel4) != ''" />
<Warning Text="The project configuration for project "%(MissingConfigLevel4.Identity)" was not specified in the solution file for the solution configuration "$(Configuration)|$(Platform)"." Code="MSB4121" HelpKeyword="MSBuild.SolutionProjectConfigurationMissing" Condition="@(MissingConfigLevel4) != ''" />
</Target>
Я загружаю файл proj и добавляю ContinueOnError = "true" ко всем задачам MSBuild и CallTarget в цели Build / Rebuild / Clean:
XmlDocument msBuildProject = new XmlDocument();
msBuildProject.Load(Tools.getFullPath(m_strBaseDir, strSolutionProjFile));
foreach (XmlNode nodeL0 in msBuildProject.ChildNodes)
{
XmlElement elementL0 = nodeL0 as XmlElement;
if ((elementL0 != null) && (String.CompareOrdinal(nodeL0.Name, "Project") == 0))
{
foreach (XmlNode nodeL1 in elementL0.ChildNodes)
{
XmlElement elementL1 = nodeL1 as XmlElement;
if ((elementL1 != null) && (String.CompareOrdinal(nodeL1.Name, "Target") == 0))
{
if ((String.CompareOrdinal(elementL1.Attributes["Name"].Value, "Build") == 0) ||
(String.CompareOrdinal(elementL1.Attributes["Name"].Value, "Rebuild") == 0) ||
(String.CompareOrdinal(elementL1.Attributes["Name"].Value, "Clean") == 0) ||
(String.CompareOrdinal(elementL1.Attributes["Name"].Value, "Publish") == 0))
{
foreach (XmlNode nodeL2 in elementL1.ChildNodes)
{
XmlElement elementL2 = nodeL2 as XmlElement;
if ((elementL2 != null) &&
((String.CompareOrdinal(nodeL2.Name, "MSBuild") == 0) ||
(String.CompareOrdinal(nodeL2.Name, "CallTarget") == 0)))
{
elementL2.SetAttribute("ContinueOnError", "true");
}
}
}
}
}
}
}
msBuildProject.Save(Tools.getFullPath(m_strBaseDir, strSolutionProjFile));
После этого я вызываю MSBuild с конфигурацией.
Это прекрасно работает, но поскольку это не очень легко, я никогда не буду отмечать это как ответ на мой вопрос: -)