我正在创建一个应用程序,它将管理外部实用程序的多个实例,为每个实例提供数据并获取结果。
但是当我为班级编写单元测试时,我遇到了一个问题。
如何测试目标方法在调用时实际启动进程(通过属性设置)?
我试过了:
我觉得发布代码和/或创建另一个exe来测试是否过度。
这是确切的方法:
public void Start()
{
if (!_isRunning) {
var startInfo = new ProcessStartInfo() {
CreateNoWindow = true,
UseShellExecute = true,
FileName = _cmdLine,
Arguments = _args
};
_process = Process.Start(startInfo);
_isRunning = true;
} else {
throw new InvalidOperationException("Process already started");
}
}
我想对它进行单元测试,以便在没有任何内容运行时(_isRunning == false),应该生成一个新进程。
我感到难过,是否有一种优雅的方式来对外部过程实际启动进行单元测试?
答案 0 :(得分:11)
我会使用依赖注入和使用模拟或假类来解决这个问题。注意我使用实例方法来启动而不是类方法。在常规代码中,您可以使用默认构造函数,它将为您创建一个流程。对于测试,您可以注入模拟或伪造进程,只需检查在模拟对象上调用正确的方法,而不必实际启动进程。您需要调整此项以考虑我省略的属性。防爆。下面:
public class UtilityManager
{
public Process UtilityProcess { get; private set; }
private bool _isRunning;
public UtilityManager() : this(null) {}
public UtilityManager( Process process )
{
this. UtilityProcess = process ?? new Process();
this._isRunning = false;
}
public void Start()
{
if (!_isRunning) {
var startInfo = new ProcessStartInfo() {
CreateNoWindow = true,
UseShellExecute = true,
FileName = _cmdLine,
Arguments = _args
};
this.UtilityProcess.Start(startInfo);
_isRunning = true;
} else {
throw new InvalidOperationException("Process already started");
}
}
测试代码......
[TestMethod]
public void StartTest()
{
Process proc = new FakeProcess(); // May need to use a wrapper class
UtilityManager manager = new UtilityManager( proc );
manager.CommandLine = "command";
...
manager.Start();
Assert.IsTrue( proc.StartCalled );
Assert.IsNotNull( proc.StartInfo );
Assert.AreEqual( "command", proc.StartInfo.FileName );
...
}