MSBuild 条件未评估

问题描述

我有一个应用程序,它可以采用 3 种“风格”之一,风格仅取决于项目中引用的 NuGet 包。 为了简化构建,我想使用自定义属性来定义风格,然后在包引用上使用条件。 应该可以使用 Visual Studio 或 MSBuild CommandLine 构建应用程序。 我将此添加到 PropertyGroup:

<Flavor Condition= "'$(Flavor)'==''">Flavor1</Flavor>

我可以看到自定义属性设置正确,只是忽略了条件 这是我试过的:

  1. 按照此处所述在 packagereference 本身上设置条件:https://docs.microsoft.com/en-us/nuget/consume-packages/package-references-in-project-files#adding-a-packagereference-condition
  2. 将条件包放在单独的 ItemGroup 中并在那里设置条件
  3. 与#2 相同,但带有选择+时间

仅#3 有效且仅在 Visual Studio 中有效 我需要一个可以在 Visual Studio 和 MSBuild 命令行中使用的解决方

#1 看起来像这样:

<packagereference Include="Falvor1Package" Condition="'$(Flavor)'=='Flavor1'">
    <Version>1.1.1.1</Version>
</packagereference>
<packagereference Include="Falvor2Package" Condition="'$(Flavor)'=='Flavor2'">
    <Version>1.1.1.1</Version>
</packagereference>

#2 看起来像这样:

<ItemGroup Condition="'$(Flavor)'=='Flavor1'">
    <packagereference Include="Falvor1Package">
        <Version>1.1.1.1</Version>
    </packagereference>
</ItemGroup>
<ItemGroup Condition="'$(Flavor)'=='Flavor2'">
    <packagereference Include="Falvor2Package">
        <Version>1.1.1.1</Version>
    </packagereference>
</ItemGroup>

#3 看起来像这样:

<Choose>
    <When Condition="'$(Flavor)'=='Flavor1'">
        <ItemGroup>
            <packagereference Include="Falvor1Package">
                <Version>1.1.1.1</Version>
            </packagereference>
        </ItemGroup>
    </When>
</Choose>
<Choose>
    <When Condition="'$(Flavor)'=='Flavor2'">
        <ItemGroup>
            <packagereference Include="Falvor2Package">
                <Version>1.1.1.1</Version>
            </packagereference>
        </ItemGroup>
    </When>
</Choose>

我使用的是 VS2019 和 MSBuild 16 我错过了什么?

解决方法

AFAIK,那行不通。您不能在 PackageReference 上使用任意条件(属性)。另见this

您可以使用条件来控制是否包含包, 其中条件可以使用任何 MSBuild 变量或定义的变量 目标或道具文件。不过,目前只有 支持 TargetFramework 变量。

换句话说,你可以做这样的事情,但不能使用你的自定义属性:

<ItemGroup>
    <PackageReference Include="..." Version="..." Condition="'$(TargetFramework)' == '...'" />
</ItemGroup>

其中 TargetFramework 的值必须是有效的 TFM

将条件添加到/a 封闭的 ItemGroup 元素时同样适用。

猜测: 我想原因是,当在这里支持任意条件时,Visual Studio 不能很好地发挥作用。因为最终,它知道引用了哪些 nuget 包(最后是程序集)。 TargetFramework 值是通过更改 Visual Studio 中的相应设置来更改的,因此它可以(手动)应对,相应地调整引用的包。对于任意的东西,很难做到这一点。 但这一切都只是猜测,可能完全错误。

,

我认为这是 VS IDE nuget Restore 上的问题。问题出在非 SDK 网络框架项目的 PackageReference nuget 管理格式上。但是,这个问题不是针对 new-sdk 项目的。

我已多次测试该问题,VS IDE nuget restore 将忽略该情况。毫无疑问,这是一个问题。

所以我建议你可以report the issue to the Team

I have raised the issue on Github.

但是nuget restoremsbuild restore 可以指定non-sdk 项目的条件。所以问题在于 msbuild restorenuget restoreVS IDE Restore 之间的区别。

所以作为一个建议,你可以对VS2019的项目做一些改动:

右键单击项目属性-->构建事件-->在预构建事件命令行上添加:

"C:\Program Files (x86)\Microsoft Visual Studio\2019\Community\MSBuild\Current\Bin\MSBuild.exe" -t:restore $(ProjectDir)

然后然后您就可以毫无问题地在 VS2019 上构建您的项目。