首页 > 解决方案 > 使用 NSubstitute 断言收到的 ILogger 调用不匹配

问题描述

ILogger当测试的方法抛出异常时,我在为调用匹配执行测试时遇到了麻烦,我真的不明白是什么原因。

这看起来像我的测试方法:

public async Task My_test_method()
{
    //Arrange

    var _logger = Substitute.For<ILogger<MyType>>();
    var _mockedInnerService = Substitute.For<IMockedInnerService>();
    var _testedService = new TestedService(_mockedInnerService, _logger);

    var errorMessage = "Some problem";
    _mockedInnerService
        .When(async _self => await _self.SomeMethod(Arg.Any<string>(), Arg.Any<string>()))
        .Do(_self => throw new Exception(errorMessage));
    var expectedMessage = $"Error: {errorMessage}";
    var methodParameters = new List<Guid>() { Guid.NewGuid() };

    //Act
    var results = await _testedService.TestedCall(methodParameters);

    //Assert
    await _mockedInnerService
        .Received(1)
        .SomeMethod(Arg.Any<string>(), Arg.Any<string>());
    results
        .Should()
        .BeEmpty();
    _logger
        .Received(1)
        .LogError(Arg.Is<string>(message => message == expectedMessage));
}

这就是我测试过的方法的样子:

public async Task<IList<SomeObject>> TestedCall(IList<Guid> ids)
{

    IList<object> results = null;
    try
    {
        units = await _innerService.SomeMethod("arg", "arg");
    }
    catch (Exception e)
    {
        var msg = "Some problem";
        _logger.LogError(msg);
    }

    return results == null
        ? new List<object>()
        : results.Select(s => new SomeObject(s.Id, s.Code)).ToList();
}

测试失败,结果如下:

Message: 
    NSubstitute.Exceptions.ReceivedCallsException : Expected to receive exactly 1 call matching:
        Log<Object>(Error, 0, [null], <null>, Func<Object, Exception, String>)
    Actually received no matching calls.
    Received 1 non-matching call (non-matching arguments indicated with '*' characters):
        Log<Object>(Error, 0, *Some problem*, <null>, Func<Object, Exception, String>)

你们能帮我理解我在这里做错了什么吗?

谢谢

标签: c#unit-testingnsubstituteilogger

解决方案


看起来该ILogger<T>.LogError(string)方法实际上是一个扩展方法。这意味着它将转换为引擎盖下的不同调用,这就是Log<Object>(Error, 0, [null], <null>, Func<Object, Exception, String>)您在输出中看到的调用。这个方法调用可以有额外的参数,每次调用扩展方法的参数可能不同,因此在你的实际调用和你的断言调用之间可能会有所不同。

要修复它,您可能必须对底层方法进行断言,并使用Arg.Any<T>()它使其忽略有问题的参数,这可能是Func<Object, Exception, String>最后的参数。


推荐阅读