背景
- 我在 Windows 7 上使用 Powershell 2.0。
- 我正在 Powershell 模块中编写一个 cmdlet(“模块”是 Powershell 2.0 的新功能)。
- 为了测试 cmdlet,我正在 Visual Studio 2008 中编写以编程方式调用 cmdlet 的单元测试。
参考
- MSDN 上的这篇名为“如何从 Cmdlet 中调用 Cmdlet”的文章展示了如何从 C# 调用 cmdlet。
源代码
这是我的实际代码的精炼版本——我已经把它做得尽可能小,这样你就可以清楚地看到我遇到的问题:
using System; using System.Management.Automation; namespace DemoCmdLet1 { class Program { static void Main(string[] args) { var cmd = new GetColorsCommand(); foreach ( var i in cmd.Invoke<string>()) { Console.WriteLine("- " + i ); } } } [Cmdlet("Get", "Colors")] public class GetColorsCommand : Cmdlet { protected override void ProcessRecord() { this.WriteObject("Hello"); this.WriteVerbose("World"); } } }
注释
- 我了解如何从 Powershell 命令行启用和捕获详细输出;那不是问题。
- 在这种情况下,我以编程方式从 C# 调用 cmdlet。
- 我发现没有任何东西可以解决我的具体情况。一些文章建议我应该实现自己的 PSHost,但似乎很昂贵,而且似乎必须将 cmdlet 作为文本调用,我想避免这种情况,因为它不是强类型的。
2009-07-20 更新
这是基于以下答案的源代码。
有些事情我仍然不清楚: * 如何调用“Get-Colors” cmdlet(理想情况下不必将其作为字符串传递给 ps 对象) * 如何在生成时获取详细输出而不是获取最后收集它们。
using System;
using System.Management.Automation;
namespace DemoCmdLet1
{
class Program
{
static void Main(string[] args)
{
var ps = System.Management.Automation.PowerShell.Create();
ps.Commands.AddScript("$verbosepreference='continue'; write-verbose 42");
foreach ( var i in ps.Invoke<string>())
{
Console.WriteLine("normal output: {0}" , i );
}
foreach (var i in ps.Streams.Verbose)
{
Console.WriteLine("verbose output: {0}" , i);
}
}
}
[Cmdlet("Get", "Colors")]
public class GetColorsCommand : Cmdlet
{
protected override void ProcessRecord()
{
this.WriteObject("Red");
this.WriteVerbose("r");
this.WriteObject("Green");
this.WriteVerbose("g");
this.WriteObject("Blue");
this.WriteVerbose("b");
}
}
}
上面的代码生成这个输出:
d:\DemoCmdLet1\DemoCmdLet1>bin\Debug\DemoCmdLet1.exe
verbose output: 42
2010-01-16 更新
通过使用 Powershell 类(在 System.Management.Automation 中找到,但仅在 powershell 2.0 SDK 附带的程序集版本中,而不是在 Windows 7 上开箱即用的版本中)我可以以编程方式调用 cmdlet 和得到详细的输出。剩下的部分是实际向该 powershell 实例添加一个自定义 cmdlet——因为这是我最初的目标——对我的 cmdlet 进行单元测试,而不是那些带有 powershell 的。
class Program
{
static void Main(string[] args)
{
var ps = System.Management.Automation.PowerShell.Create();
ps.AddCommand("Get-Process");
ps.AddParameter("Verbose");
ps.Streams.Verbose.DataAdded += Verbose_DataAdded;
foreach (PSObject result in ps.Invoke())
{
Console.WriteLine(
"output: {0,-24}{1}",
result.Members["ProcessName"].Value,
result.Members["Id"].Value);
}
Console.ReadKey();
}
static void Verbose_DataAdded(object sender, DataAddedEventArgs e)
{
Console.WriteLine( "verbose output: {0}", e.Index);
}
}
[Cmdlet("Get", "Colors")]
public class GetColorsCommand : Cmdlet
{
protected override void ProcessRecord()
{
this.WriteObject("Hello");
this.WriteVerbose("World");
}
}