就是这种情况。我有异步调用所以我需要为此做一个Mid tier才能测试它。
request.BeginGetResponse(new AsyncCallback(LoginCallback), requestState);
因此,为了能够在没有实际请求的情况下进行测试,我创建了可以模拟的界面。
public interface IRequestSender
{
void Send(HttpWebRequest request, AsyncCallback internalCallback, object requestState);
}
然后在实现中我可以使用上面的那个调用,我可以提供一些模拟类来调用我的回调方法,无论请求是否有效。我的模拟课看起来像这样。
public class RequestSenderMock : IRequestSender
{
public void Send(HttpWebRequest request, AsyncCallback internalCallback, object requestState)
{
var result = new Mock<IAsyncResult>();
result.Setup(x => x.AsyncState).Returns(requestState);
internalCallback(result.Object);
}
}
我现在可以在单元测试中轻松创建模拟对象并使用它。但是当我创建
时var sender = new Mock<RequestSenderMock>();
我无法验证此对象的通话计数。
sender.Verify(x => x.Send(It.IsAny<HttpWebRequest>(), It.IsAny<AsyncCallback>(), It.IsAny<object>()), Times.Once());
它说我的方法需要是虚拟的。 有没有办法在不使我的方法虚拟的情况下做到这一点?如果我能在某种程度上指定使用接口时的方法实现,那将是最好的。
var sender = new Mock<IRequestSender>();
以某种方式使用Setup方法或其他一些方法在这个模拟对象上进行实现。比我简单地删除我的模拟课程。 这可能吗?你有什么建议?
答案 0 :(得分:10)
我发现您正在创建一个手动模拟并使用模拟框架来模拟它(模拟模拟),这让我感到困惑。我会考虑将您的自定义函数移动到某个实用程序类中,并使用回调代替。
示例:
public class RequestSenderHelpers
{
public static void Send(HttpWebRequest request, AsyncCallback internalCallback, object requestState)
{
var result = new Mock<IAsyncResult>();
result.Setup(x => x.AsyncState).Returns(requestState);
internalCallback(result.Object);
}
}
[Test]
public void Callback_VerifyingWithMethodImplementation_VerifyWorks()
{
// arrange
var sender = new Mock<IRequestSender>();
sender.Setup(s => s.Send(It.IsAny<HttpWebRequest>(), It.IsAny<AsyncCallback>(), It.IsAny<object>())).Callback<HttpWebRequest, AsyncCallback, object>(RequestSenderHelpers.Send);
// act
sender.Object.Send(null, delegate {}, null);
// assert
sender.Verify(s => s.Send(It.IsAny<HttpWebRequest>(), It.IsAny<AsyncCallback>(), It.IsAny<object>()));
}
为了避免冗长的设置,您可以在扩展方法中包装方法的设置并相应地更改测试:
public static class RequestSenderHelpers
{
public static void Send(HttpWebRequest request, AsyncCallback internalCallback, object requestState)
{
var result = new Mock<IAsyncResult>();
result.Setup(x => x.AsyncState).Returns(requestState);
internalCallback(result.Object);
}
public static void SetupSendWithMockedAsyncResult(this Mock<IRequestSender> sender)
{
sender.Setup(s => s.Send(It.IsAny<HttpWebRequest>(), It.IsAny<AsyncCallback>(), It.IsAny<object>())).Callback<HttpWebRequest, AsyncCallback, object>(Send);
}
}
[Test]
public void Callback_VerifyingWithMethodImplementation_VerifyWorks()
{
// arrange
var sender = new Mock<IRequestSender>();
sender.SetupSendWithMockedAsyncResult();
// act
sender.Object.Send(null, delegate {}, null);
// assert
sender.Verify(s => s.Send(It.IsAny<HttpWebRequest>(), It.IsAny<AsyncCallback>(), It.IsAny<object>()));
}