MSBuild ReadLinesFromFile一行上的所有文本

时间:2008-11-07 15:54:01

标签: deployment msbuild msbuild-task msbuildcommunitytasks

当我在MSBUILD中的文件上执行ReadLinesFromFile并再次输出该文件时,我将所有文本放在一行上。所有Carriage return和LineFeeds都被删除了。

<Project DefaultTargets = "Deploy"
xmlns="http://schemas.microsoft.com/developer/msbuild/2003" >
<Import  Project="$(MSBuildExtensionsPath)\MSBuildCommunityTasks\MSBuild.Community.Tasks.Targets"/>

<ItemGroup>
    <MyTextFile Include="$(ReleaseNotesDir)$(NewBuildNumber).txt"/>
</ItemGroup>

<Target Name="ReadReleaseNotes">
    <ReadLinesFromFile
        File="@(MyTextFile)" >
        <Output
            TaskParameter="Lines"
            ItemName="ReleaseNoteItems"/>
    </ReadLinesFromFile>
</Target>

<Target Name="MailUsers" DependsOnTargets="ReadReleaseNotes" >
    <Mail SmtpServer="$(MailServer)"
        To="$(MyEMail)"
        From="$(MyEMail)"
        Subject="Test Mail Task"
        Body="@(ReleaseNoteItems)" />
</Target>
<Target Name="Deploy">
    <CallTarget Targets="MailUsers" />
</Target>

</Project>

我从文件中获取通常看起来像这样的文本

- New Deployment Tool for BLAH

- Random other stuff()""

像这样出来

- New Deployment Tool for BLAH;- Random other stuff()""

我知道ReadLinesFromFile的代码会一次一行地提取数据并去除回车。

有没有办法让它们重新进入? 所以我的电子邮件看起来格式很好?

谢谢

4 个答案:

答案 0 :(得分:41)

这里的问题是你以一种不想要的方式使用ReadLinesFromFile任务。

  

ReadLinesFromFile任务
  从文本文件中读取项目的列表。

因此,它不只是从文件中读取所有文本,而是从文件中读取单个项目并返回ITaskItems的项目组。每当使用@()语法输出项目列表时,您将获得一个单独的列表,默认值为分号。此示例说明了此行为:

<?xml version="1.0" encoding="utf-8"?>
<Project DefaultTargets="Build"    xmlns="http://schemas.microsoft.com/developer/msbuild/2003" ToolsVersion="3.5">

    <ItemGroup>
        <Color Include="Red" />
        <Color Include="Blue" />
        <Color Include="Green" />
</ItemGroup>

<Target Name="Build">
        <Message Text="ItemGroup Color: @(Color)" />
</Target>

</Project>

输出如下:

  ItemGroup Color: Red;Blue;Green

因此,虽然解决问题的最佳方法是编写一个MSBuild任务,将文件作为字符串读入属性,但不是项目列表,这实际上不是您要求的。您询问是否有方式将它们放回去,并且使用了MSBuild Transforms

转换用于从另一个列表创建一个列表,并且还可以使用自定义分隔符进行转换。所以答案是将使用ReadItemsFromFile读入的列表转换为另一个带换行符的列表。这是一个例子:

<?xml version="1.0" encoding="utf-8"?>
<Project DefaultTargets="Build" xmlns="http://schemas.microsoft.com/developer/msbuild/2003" ToolsVersion="3.5">

    <ItemGroup>
        <File Include="$(MSBuildProjectDirectory)\Test.txt" />
    </ItemGroup>

    <Target Name="Build">
        <ReadLinesFromFile File="@(File)">
            <Output TaskParameter="Lines" ItemName="FileContents" />
        </ReadLinesFromFile>

        <Message Text="FileContents: @(FileContents)" />
        <Message Text="FileContents Transformed: @(FileContents->'%(Identity)', '%0a%0d')" />
    </Target>

</Project>

Test.text看起来像:

Red
Green
Blue

输出如下:

[C:\temp]:: msbuild test.proj
Microsoft (R) Build Engine Version 3.5.21022.8
[Microsoft .NET Framework, Version 2.0.50727.1433]
Copyright (C) Microsoft Corporation 2007. All rights reserved.

Build started 11/8/2008 8:16:59 AM.
Project "C:\temp\test.proj" on node 0 (default targets).
  FileContents: Red;Green;Blue
  FileContents Transformed: Red
Green
Blue
Done Building Project "C:\temp\test.proj" (default targets).


Build succeeded.
    0 Warning(s)
    0 Error(s)

Time Elapsed 00:00:00.03

这里发生了两件事。

@(FileContents->'%(Identity)', '%0a%0d')   
  • 我们正在使用相同的值(Identity)将列表从一种类型转换为另一种类型,但是使用自定义分隔符'%0a%0d'
  • 我们正在使用MSBuild Escaping转义换行符(%0a)和回车符(%0d

答案 1 :(得分:35)

如果您使用的是MSBuild 4.0,则可以执行以下操作,以获取文件的内容:

$([System.IO.File]::ReadAllText($FilePath))

答案 2 :(得分:4)

而不是@(FileContents-&gt;'%(Identity)','%0a%0d')我相信你可以做@(FileContents,'%0a%0d')

答案 3 :(得分:1)

您可以将WriteLinesToFile与

结合使用
$([System.IO.File]::ReadAllText($(SourceFilePath))):

< WriteLinesToFile File="$(DestinationFilePath)"  Lines="$([System.IO.File]::ReadAllText($(SourceFilePath)))"
                      Overwrite="true" 
                      />

这将完全复制您的文件。