我正在使用NUnit构建C#单元测试,并且想要测试主程序根据命令行参数实际输出正确的输出。
在调用Program.Main(...)
的NUnit测试方法中,是否有一种方式可以抓取写入到Console.Out
和Console.Error
的所有内容,以便我可以进行验证?
我正在使用NUnit构建C#单元测试,并且想要测试主程序根据命令行参数实际输出正确的输出。
在调用Program.Main(...)
的NUnit测试方法中,是否有一种方式可以抓取写入到Console.Out
和Console.Error
的所有内容,以便我可以进行验证?
你可以将 Console.In
、Console.Out
和 Console.Error
重定向到自定义的StringWriter,像这样:
[TestMethod]
public void ValidateConsoleOutput()
{
using (StringWriter sw = new StringWriter())
{
Console.SetOut(sw);
ConsoleUser cu = new ConsoleUser();
cu.DoWork();
string expected = string.Format("Ploeh{0}", Environment.NewLine);
Assert.AreEqual<string>(expected, sw.ToString());
}
}
有关详细信息,请参见此博客文章。
Console.SetOut(new StreamWriter(Console.OpenStandardError())
(您可能还需要将Autoflush
设置为true
)来重置标准输出。这样做后,它将适用于任何测试运行器,包括R#。 - Abel您可以使用此简单类来使用“using”语句获取输出:
public class ConsoleOutput : IDisposable
{
private StringWriter stringWriter;
private TextWriter originalOutput;
public ConsoleOutput()
{
stringWriter = new StringWriter();
originalOutput = Console.Out;
Console.SetOut(stringWriter);
}
public string GetOuput()
{
return stringWriter.ToString();
}
public void Dispose()
{
Console.SetOut(originalOutput);
stringWriter.Dispose();
}
}
以下是如何使用它的示例:
using (var consoleOutput = new ConsoleOutput())
{
target.WriteToConsole(text);
Assert.AreEqual(text, consoleOutput.GetOuput());
}
你可以在我的博客文章中找到更详细的信息和可用的代码示例 - 在单元测试中获取控制台输出.
你仍然需要一个集成测试,而不是单元测试,正如其他人正确建议的那样。
ProgramTest.cs
using NUnit.Framework;
class ConsoleTests
{
[Test]
public void TestsStdOut()
{
var capturedStdOut = CapturedStdOut(() =>
{
RunApp();
});
Assert.AreEqual("Welcome, John!", capturedStdOut);
}
void RunApp(string[]? arguments = default)
{
var entryPoint = typeof(Program).Assembly.EntryPoint!;
entryPoint.Invoke(null, new object[] { arguments ?? Array.Empty<string>() });
}
string CapturedStdOut(Action callback)
{
TextWriter originalStdOut = Console.Out;
using var newStdOut = new StringWriter();
Console.SetOut(newStdOut);
callback.Invoke();
var capturedOutput = newStdOut.ToString();
Console.SetOut(originalStdOut);
return capturedOutput;
}
}
Program.cs
Console.Write($"Welcome, John!");