使用 NSubstitute 在单元测试中验证 Expression<Func<T, bool>> 参数

Verify Expression<Func<T, bool>> argument in a unit test with NSubstitute

我正在进行单元测试,我想验证模拟对象是否收到了正确的谓词参数。但是,我没能成功。

这是代码的框架:

public interface IRepository<T>
{
    Task<T> SingleOrDefaultAsync(Expression<Func<T, bool>> predicate);   
}

public class MyClass
{
    private readonly IRepository<Foo> _repository

    public MyClass(IRepository<Foo> repository)
    {
        _repository = repository;
    }

    public Task<bool> MyMethod(int id)
    {
        var foo = _repository.SingleOrDefaultAsync(x => x.Id == id);

        return foo != null;
    }
}

在我的测试中class我有以下方法

public async Task MyTestMethod()
{
    var repository = Substitute.For<IRepository<Foo>>();
    repository.SingleOrDefaultAsync(x => x.Id == 123).Returns(Task.FromResult(new Foo()));

    var myClass = new MyClass(repository);

    var result = await myClass.MyMethod(123);

    result.Should().BeTrue();
}

但是如上所述,这个测试失败了。我可以使用 Arg.Any<Expression<Func<Foo, bool>> 使其通过,但感觉不对。

有人对我做错了什么提出建议吗?

捕获传递给模拟的表达式并在 Returns 中使用它来验证预期的行为。

例如

public async Task MyTestMethod() {
    //Arrange
    var id = 123;
    var model = new Foo() {
        Id = id;
    }

    var repository = Substitute.For<IRepository<Foo>>();

    repository.SingleOrDefaultAsync(Arg.Any<Expression<Func<Foo, bool>>())
        .Returns(args => {
            var expression = args.Arg<Expression<Func<Foo, bool>>(); //capture expression
            Foo result = expression.Compile()(model) ? model : null; //use to verify behavior
            Task.FromResult(result);
        });

    var myClass = new MyClass(repository);

    //Act
    var actual = await myClass.MyMethod(id);

    //Assert
    actual.Should().BeTrue();
}