【问题标题】:MSBuild: Deploying files that are not included in the projectMSBuild:部署项目中不包含的文件
【发布时间】:2012-10-02 20:46:51
【问题描述】:

我在一个 Web 项目上有一个 Pre-build 事件,它使用 node.js 缩小和连接 javascript 文件。这将在脚本文件夹中创建一个名为 BuiltScripts 的文件夹,该文件夹是脚本文件夹的副本,但文件已被缩小。当我进行部署时,我想发布脚本文件夹,包括其中的 BuiltScripts 文件夹。为此,我已将 BuiltScripts 文件夹添加到项目中。这不是一个理想的解决方案:

  1. 我必须检查 BuiltScripts 文件夹才能构建,因为其中的文件是只读的,因为解决方案受源代码控制。这会在签入时造成麻烦,因为我签出的文件太多。
  2. 当我向项目添加新文件时,我必须确保记得将其添加到 BuiltScripts 文件夹,否则将不会部署文件的构建版本。
  3. 我的构建将在构建服务器上失败,因为 BuiltScripts 文件夹中的文件也是只读的。
  4. 在搜索文件和进行基于文本的搜索时,拥有两个同名文件副本是一个问题。

我希望构建服务器构建并缩小 javascript 文件作为预构建步骤,但我不希望将 BuiltScripts 文件夹添加到项目中。但是,当构建服务器最后打包项目时,我希望它复制带有构建过程输出的 BuiltScripts 文件夹。我怎样才能做到这一点?

【问题讨论】:

  • 你使用什么来部署\发布 - 这是标准的 msbuild 目标还是你定制的东西?
  • @AlexeyShcherbak MSBuild,TFS,没有自定义
  • 我怀疑 Sayed Ibrahim Hashimi 在您提出问题后写了这篇文章并给出了答案:)。 sedodream.com/2012/10/09/…
  • =) 好的,这可能纯属巧合。无论如何,看起来他的答案最适合您的问题。对于这个问题,我做了一些 msbuild 挖掘与 Sayed 相同的方向。总体答案必须与他的非常相似。现在它有点过时了;)
  • @AlexeyShcherbak 为什么你说它已经过时了?

标签: deployment msbuild pre-build-event


【解决方案1】:

不要使用项目属性的预构建事件(我认为这就是您的意思),而是覆盖 .csproj/.vbproj 文件中的 BeforeBuild 目标。

<Project ...>
  ...
  <Target Name="BeforeBuild">
    <!-- Create the 'BuildScripts' directory. -->
    <!-- The $(IntermediateOutputPath) reference the 'obj' folder, which I like to -->
    <!-- use for these kinds of things. -->
    <MakeDir Directories="$(IntermediateOutputPath)BuiltScripts">
      <Output PropertyName="BuildScriptsPath" TaskParameter="DirectoriesCreated" />
    </MakeDir>
    <!-- Execute the javascript minifier. -->
    <Exec Command="..." />
    <!-- Create an item group for the minified scripts so we manipulate the target path. -->
    <CreateItem Include="$(BuildScriptsPath)\*.js">
      <Output ItemName="BuiltScripts" TaskParameter="Include" />
      <Output ItemName="FileWrites" TaskParameter="Include" />
    </CreateItem>
    <!-- Add the minified scripts to the Content item group, -->
    <!-- which the deployment MSBuild inspects for files to deploy. -->
    <CreateItem Include="@(BuiltScripts)"
      AdditionalMetadata="TargetPath=scripts\%(Filename)%(Extension)">
      <Output ItemName="ContentWithTargetPath" TaskParameter="Include" />
    </CreateItem>
  </Target>
  ...
</Project>

如果文件未部署到正确的目录,您可能必须使用作为 CreateItem 任务输出的内容项组的形状。请注意,我使用了项目转换来制作目标路径scripts\YourScript.js

另请注意,第一个 CreateItem 任务将输出填充到名为“FileWrites”的项目组中。我发现 Clean 目标会检查该项目组以了解要删除哪些文件。

将该 XML 放入您的项目文件中 &lt;Import&gt; 元素之后,您应该一切顺利。无需签入源代码控制,甚至您的构建服务器也会很高兴。

【讨论】:

  • 您会添加一个步骤来删除以前的 BuiltScripts 文件夹内容吗?
  • 我运行了我的示例代码,看起来 IncrementalClean 目标会自动删除文件,因为它们包含在 FileWrites 项目组中,这在您删除 .js 文件时很有用希望它不再被部署。
【解决方案2】:

您不能使用构建后事件来运行脚本来执行您想要的操作吗?我使用一个将我的输出文件移动到与项目完全无关的位置。无论是普通批处理、VBScript/JScript 还是 PowerShell,您几乎可以做任何您想做的事情。

【讨论】:

    【解决方案3】:

    我也有类似的需求,尽管它只涉及将现有文件添加到部署包中,而不是将它们作为构建的一部分生成。我发现this page 上的技术很有用。

    【讨论】:

      猜你喜欢
      • 2011-10-25
      • 1970-01-01
      • 1970-01-01
      • 2018-09-07
      • 1970-01-01
      • 1970-01-01
      • 1970-01-01
      • 1970-01-01
      • 1970-01-01
      相关资源
      最近更新 更多