我想模拟IPrincipal,所以我这样做了
public Mock<IPrincipal> Principal { get; set; }在我的nunit设置中
Principal = new Mock<IPrincipal>();所以这应该就是我在nunit单元测试中所需要的了,但是在我的实际控制器文件中呢?
比如我该怎么设置它?
例如,我有一个membership.Provider
所以我所做的是在我的控制器构造函数中
Provider = Membership.Provider;因此,在我的控制器中,我只使用了Provider。(我需要什么)。
我不确定如何以同样的方式设置主要的东西。
发布于 2009-08-21 22:26:23
你是在说ASP.NET MVC吗?可能是吧。
您必须创建控制器的一个实例并设置其RequestContext。模拟RequestContext的HttpContext,在此HttpContext中模拟其用户属性,并将其设置为模拟的IPrincipal:
var principal = new Moq.Mock<IPrincipal>();
// ... mock IPrincipal as you wish
var httpContext = new Moq.Mock<HttpContextBase>();
httpContext.Setup(x => x.User).Returns(principal.Object);
// ... mock other httpContext's properties, methods, as needed
var reqContext = new RequestContext(httpContext.Object, new RouteData());
// now create the controller:
var controller = new MyController();
controller.ControllerContext =
new ControllerContext(reqContext, controller);希望这能有所帮助。
编辑:
仅供参考,控制器类上的User属性来自HttpContext对象,您可以在此处看到(这是User属性的getter方法,从Reflector获得--您也可以下载ASP.NET MVC源代码):
public IPrincipal User
{
get
{
if (this.HttpContext != null)
{
return this.HttpContext.User;
}
return null;
}
}如果现在检查HttpContext属性,您将看到:
public HttpContextBase HttpContext
{
get
{
if (base.ControllerContext != null)
{
return base.ControllerContext.HttpContext;
}
return null;
}
}因此,到目前为止,所有内容都是“只读”的。我们需要一种方法来“注入”一个被嘲笑的“用户”。因此,我们检查是否可以通过属性在控制器上实际注入ControllerContext对象。我们验证它是如何获得其"HttpContext“对象的,以了解如何正确地模拟它:
public virtual HttpContextBase HttpContext
{
get
{
if (this._httpContext == null)
{
this._httpContext = (this._requestContext != null) ? this._requestContext.HttpContext : new EmptyHttpContext();
}
return this._httpContext;
}
set
{
this._httpContext = value;
}
}因此,在这里我们可以看到,ControllerContext对象从一个RequestContext对象获取它的HttpContext。因此,这可能解释了我上面所做的:
在所有这些魔术之后,控制器将不会知道您在没有通过Web服务器建立实际连接的情况下调用它。
因此,您可以继续像往常一样使用控制器中的"User“属性,不需要做任何更改。
https://stackoverflow.com/questions/1314370
复制相似问题