11

我正在使用 Moq 来验证是否在我的单元测试中调用了一个方法。在这种特定情况下,我想测试被测方法是否通过 log4net 记录错误。问题是,这可以通过调用log.Error或来完成log.ErrorFormat。哪一个都好。

我该如何验证呢?我只知道如何验证他们都被调用了。

var logMock = new Mock<ILog>();

var myClass = new MyClass(logMock.Object);

myClass.MyMethod();

logMock.Verify(log => log.Error(It.IsAny<object>()));
logMock.Verify(log => log.ErrorFormat(It.IsAny<string>(), It.IsAny<object>()));

现在我想起来了,它们都有一堆重载,我不介意是否调用了任何重载(我开始怀疑这是一个很好的测试)。

提前致谢。

编辑:我只是想到了一些讨厌的东西:

try
{
    logMock.Verify(log => log.Error(It.IsAny<object>()));
}
catch (Moq.MockException ex)
{
    logMock.Verify(log => log.ErrorFormat(It.IsAny<string>(), It.IsAny<object>()));
}

也许我可以用某种扩展方法包装它......例如VerifyAny

4

2 回答 2

14

您可以为每个设置标志的有效错误方法注册一个回调:

// Arrange
bool errorFlag = false;
logMock
    .Setup(l => l.Error(It.IsAny<object>()))
    .Callback((object o) => errorFlag = true);

/* repeat setup for each logMock method */

// Act
myClass.MyMethod();

// Assert
Assert.IsTrue(errorFlag);

当然,如果您要覆盖许多重载,这仍然会很乏味。

编辑:为了好玩,这里有一个扩展方法Mock<T>.VerifyAny

public static class MockExtensions
{
    public static void VerifyAny<T>(this Mock<T> mock, params Expression<Action<T>>[] expressions)
        where T: class
    {
        List<MockException> exceptions = new List<MockException>();
        bool success = false;
        foreach (var expression in expressions)
        {
            try
            {
                mock.Verify(expression);
                success = true;
                break;
            }
            catch (MockException ex)
            {
                exceptions.Add(ex);
            }
        }

        if (!success)
        {
            throw new AggregateException("None of the specified methods were invoked.", exceptions);
        }
    }
}

用法:

[TestMethod]
public void FooTest()
{
    Mock<IFoo> fooMock = new Mock<IFoo>();
    fooMock.Object.Bar1();

    fooMock.VerifyAny(
        f => f.Bar1(),
        f => f.Bar2());
}
于 2013-01-10T00:20:04.057 回答
1

如果您正在专门测试是否记录了特定错误,为什么不进行 2 个测试,一个确保log.Error被调用,一个确保log.ErrorFormat被调用,我假设您可以根据输入控制调用哪一个。

如果您仍然想验证其中一个,您可以使用这种方法,它完全符合您的需要:

验证单元测试中是否调用了一种方法或另一种方法

于 2013-01-10T00:24:39.743 回答