2

我似乎无法使用Moq来模拟HttpContext.Request.Url.Authority,因为它是一种非虚拟方法。我得到以下异常:

{"Invalid setup on a non-virtual (overridable in VB) member: p => p.HttpContext.Request.Url.Authority"}

我该如何克服呢?以下是我的测试方法:

[TestMethod]
public void ForgottenPasswordPost_Requested_CaptchaCorrectEmailExists()
{
    _testModel.ControllerContext.SetupGet(p => p.HttpContext.Session["Captcha"]).Returns("HelloWorld");
    _testModel.ControllerContext.SetupGet(p => p.HttpContext.Request.Url.Authority).Returns("www.localhost.com");
    _testModel.QMember.Setup(m => m.MemberExistsWithEmail(It.IsAny<string>())).Returns(true);


    var controllerUnderTest = _testModel.ReturnController();
    ForgottenPasswordModel model = new ForgottenPasswordModel() { Captcha = "HelloWorld" };

    //Act
    var actionResult = (RedirectToRouteResult)controllerUnderTest.ForgottenPassword(model);

    Assert.AreEqual("ForgottenPasswordConfirm", actionResult.RouteValues["action"]);
    Assert.AreEqual("a", actionResult.RouteValues["controller"]);
}

public class TestModel
{
    public UnregisteredController Controller { get; set; }
    public Mock<ControllerContext> ControllerContext { get; set; }
    public Mock<IQ_Member> QMember { get; set; }

    public TestModel()
    {
        ControllerContext = new Mock<ControllerContext>();
        QMember = new Mock<IQ_Member>();
    }

    public UnregisteredController ReturnController()
    {
        Controller = new UnregisteredController(QMember.Object);
        Controller.ControllerContext = ControllerContext.Object;
        return Controller;
    }
}
4

1 回答 1

4

根据此 SO post,您只能使用 Moq 实现虚拟/抽象成员。

在一般非 ASP.NET MVC 情况下,您可以

  1. 使用允许覆盖非虚拟的、具体的类成员的模拟工具或

  2. 编写您自己的包装类,将传递方法传递给静态/非虚拟成员,从这些包装类中提取接口,然后设计其余代码以依赖这些接口,然后可以轻松地模拟和注入依赖项。

我会推荐 #2 而不是上面的 #1,即使它会产生大量的工作,因为 #2 几乎肯定会导致代码更好地遵守Dependency Inversion 原则

但是,在您的情况下,我认为有一个相当简单的解决方案。

在代码行中

_testModel.ControllerContext.SetupGet(p => p.HttpContext.Request.Url.Authority).Returns("www.localhost.com");
  • .HttpContext是抽象的,所以可以模拟
  • .Request是抽象的,所以可以模拟
  • .Url是虚拟的,所以可以模拟
  • .Authority是一个非虚拟字符串属性,这就是给你带来问题的原因

试试这个,因为这.Url是你可以模拟的最深的部分:

_testModel.ControllerContext.SetupGet(p => p.HttpContext.Request.Url).Returns(new Uri("http://www.localhost.com"));
于 2013-04-19T21:29:40.390 回答