我有一个很好的power-shell驱动的后期构建脚本,它可以完成一些神奇的mumbo-jumbo,用于在编译完所有内容之后将所有插件拉到正确的位置,并使用其中的主项目来整理它们的依赖项。溶液
我的问题是,偶尔当我做一些愚蠢的事情时,我最终会处于一种状态,即我的脚本无法正确执行它的操作,并在powershell中抛出一个异常,其中包含错误的详细信息。
有没有办法让这些异常进入Visual Studio Error窗口,这样当我的后期构建失败时,vstudio会收到失败通知,并在窗口中将其与所有其他警告/错误一起很好地格式化?< / p>
编辑:这是对我正在寻找的内容的澄清。
删除了死亡的ImageShack链接
注意: 如果你已经登陆这里寻找如何在Visual Studio中显示自定义错误消息(当你控制错误消息时) )您可以看到Roy Tinker's Answer了解如何定制要显示的消息。如果您有兴趣捕获无法控制的意外错误,或找到更完整的解决方案,请参阅接受的答案。
答案 0 :(得分:16)
要创建将出现在“错误列表”窗口中的错误,警告或消息,只需从构建前或构建后事件启动的脚本中以下列格式将消息记录到stdout或stderr。如果有官方规格,请评论或编辑;这只是我能够通过反复试验和观察MSBuild的输出来推断的。方括号表示“可选”:
FilePath[(LineNumber[,ColumnNumber])]: MessageType[ MessageCode]: Description
举个例子:
E:\Projects\SampleProject\Program.cs(18,5): error CS1519: Invalid token '}' in class, struct, or interface member declaration
有关更多示例,请参阅“输出”窗口,以获取在Visual Studio中运行构建时可能出现的任何错误/警告/消息。
答案 1 :(得分:10)
以下是两种在PowerShell脚本出错时中断构建的方法。
exit()
终止PowerShell进程要从脚本返回状态代码,如果非零,将显示在错误列表中,请使用以下命令:
exit(45) # or any other non-zero number, for that matter.
这并没有将错误文本准确地输入到您的错误列表中,但它会终止脚本并在错误列表中获取一些内容以指示构建前或构建后命令失败。
我花了一点时间研究在MSBuild任务中执行PowerShell脚本的细节。我有a full article with sample code on my blog。请查看它,因为它包括更多的讨论,以及如何处理示例项目的一些解释。它可能不是一个完整或完美的解决方案,但是我用一个非常简单的脚本在我的机器上工作 TM 。
此方法在报告PowerShell错误时提供行和列精度,甚至支持我们在Visual Studio中习惯的双击 - 采取我的文件行为。如果它缺乏,我相信你能够扩展它以满足你的需求。此外,根据您的Visual Studio版本,您可能需要按下诸如程序集参考版本之类的详细信息。
首先,在类库项目中构建自定义MSBuild Task。该库应引用以下程序集以进行MSBuild和PowerShell集成。 (请注意,此示例需要PowerShell 2.0。)
构建一个任务类,并公开一个属性以指定PowerShell脚本的路径,如下所示:
using System.IO;
using System.Management.Automation;
using System.Management.Automation.Runspaces;
using Microsoft.Build.Framework;
using Microsoft.Build.Utilities;
public class PsBuildTask : Task
{
[Required]
public string ScriptPath { get; set; }
public override bool Execute()
{
// ...
}
}
在Execute()
方法中,启动PowerShell运行时,执行脚本并收集错误。使用Log
属性记录错误。完成后,关闭运行空间,如果脚本没有记录错误,则返回true。
// create Powershell runspace
Runspace runspace = RunspaceFactory.CreateRunspace();
runspace.Open();
// create a pipeline and feed it the script text
Pipeline pipeline = runspace.CreatePipeline();
pipeline.Commands.AddScript(". " + ScriptPath);
// execute the script and extract errors
pipeline.Invoke();
var errors = pipeline.Error;
// log an MSBuild error for each error.
foreach (PSObject error in errors.Read(errors.Count))
{
var invocationInfo = ((ErrorRecord)(error.BaseObject)).InvocationInfo;
Log.LogError(
"Script",
string.Empty,
string.Empty,
new FileInfo(ScriptPath).FullName,
invocationInfo.ScriptLineNumber,
invocationInfo.OffsetInLine,
0,
0,
error.ToString());
}
// close the runspace
runspace.Close();
return !Log.HasLoggedErrors;
就是这样。有了这个程序集,我们可以配置另一个项目来使用MSBuild任务。
例如,考虑一个基于C#的类库项目(.csproj)。在post build事件中集成任务只需要几件事。
首先,在.csproj文件的<Project>
节点内注册任务,如下所示:
<UsingTask TaskName="PsBuildTask"
AssemblyFile="..\Noc.PsBuild\bin\Debug\Noc.PsBuild.dll" />
TaskName应该是任务类的名称,尽管看起来不需要命名空间。 AssemblyFile
是自定义MSBuild任务程序集的绝对路径,或相对于.csproj文件的相对路径。对于GAC中的程序集,您可以改为使用AssemblyName
属性。
注册后,任务可以在构建前和构建后的事件中使用。在.csproj文件的<Project>
元素中配置构建事件,如下所示:
<Target Name="AfterBuild">
<PsBuildTask ScriptPath=".\script.ps1" />
</Target>
就是这样。当Visual Studio编译项目时,它会加载自定义程序集和任务对象并执行任务。检索并报告管道引发的错误。