当我在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的代码会一次一行地提取数据并去除回车。
有没有办法让它们重新进入? 所以我的电子邮件看起来格式很好?
谢谢
答案 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'
%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"
/>
这将完全复制您的文件。