异步单元测试中的Stubbing Task返回方法

Wou*_*ort 26 c# unit-testing task-parallel-library async-await

假设我有以下类和它依赖的接口:

public class MyController
{
    private IRepository _repository;
    public MyController(IRepository repository)
    {
        _repository = repository;
    }

    public async Task MethodUnderTest(int someId)
    {
        var o = await _repository.FindById(someId);
        // update o
        await _repository.Commit();
    }
}

public interface IRepository
{
    Task Commit();
}
Run Code Online (Sandbox Code Playgroud)

当我对这个方法进行单元测试时,我可以执行以下操作(使用xUnit和Rhino Mocks):

[Fact]
public async Task MyTest()
{
    IRepository repositoryStub = MockRepository.GenerateStub<IRepository>();

    MyController controller = new MyController(repositoryStub);

    await controller.MethodUnderTest(1);
}
Run Code Online (Sandbox Code Playgroud)

这会因System.NullReferenceException而失败:对象引用未设置为对象的实例.

使用以下StackTrace:

UnitTest.MyController.<MethodUnderTest>d__0.MoveNext() in 
\UnitTest\Class1.cs:line 35
--- End of stack trace from previous location where exception was thrown ---
   at System.Runtime.CompilerServices.TaskAwaiter.ThrowForNonSuccess(Task task)
   at System.Runtime.CompilerServices.TaskAwaiter.HandleNonSuccessAndDebuggerNotification(Task task)
   at System.Runtime.CompilerServices.TaskAwaiter.GetResult()
   at \UnitTest\Class1.cs:line 20
Run Code Online (Sandbox Code Playgroud)

它是正确的,因为出现这种错误Commit()的回报null和对的statemachine异步/等待调用生成MoveNext()一个null

我可以通过以下方式解决这个问题:

repositoryStub.Expect(r => r.Commit()).Return(Task.FromResult<bool>(true);
Run Code Online (Sandbox Code Playgroud)

但这感觉有点奇怪..我可以使用任何T,FromResult<T>测试将运行.我找不到一个FromResult会返回非泛型的方法Task.

我用的是T什么关系?或者我应该以其他方式解决这个问题?

Ste*_*ary 45

你需要归还一些东西; async Task方法永远不会回来null.

T无所谓.如果需要,可以将a声明static readonly Task SuccessTask = Task.FromResult<object>(null);为辅助常量.我的AsyncEx库中有类似的常量.


Mar*_*son 6

如果您的目标是 .NET Framework 4.6 或更高版本,您现在可以.Returns(Task.CompletedTask)改用稍短的版本。