3

我正在测试一个调用此类方法的 MVC 3 控制器:

public class SessionVar
{
    /// <summary>
    /// Gets the session.
    /// </summary>
    private static HttpSessionState Session
    {
        get
        {
            if (HttpContext.Current == null)
                throw new ApplicationException
                                   ("No Http Context, No Session to Get!");

            return HttpContext.Current.Session;
        }
    }

    public static T Get<T>(string key)
    {
        return Session[key] == null ? default(T) : (T)Session[key];
    }
    ...
}

根据Hanselman 博客的建议,我的测试方法是:

[Test]
public void CanRenderEmployeeList()
{
    _mockIEmployeeService.Setup(s => s.GetEmployees(StatusFilter.OnlyActive))
        .Returns(BuildsEmployeeList().Where(e => e.IsApproved));

    var httpContext = FakeHttpContext();
    var target = _employeeController;
    target.ControllerContext = new ControllerContext
                  (new RequestContext(httpContext, new RouteData()), target);
    var result = target.Index();

    Assert.IsNotNull(result);
    Assert.IsInstanceOf<ViewResult>(result);
    var viewModel = target.ViewData.Model;
    Assert.IsInstanceOf<EmployeeListViewModel>(viewModel);
}

public static HttpContextBase FakeHttpContext()
{
    var context = new Mock<HttpContextBase>();
    var request = new Mock<HttpRequestBase>();
    var response = new Mock<HttpResponseBase>();
    var session = new Mock<HttpSessionStateBase>();
    var server = new Mock<HttpServerUtilityBase>();

    context.Setup(ctx => ctx.Request).Returns(request.Object);
    context.Setup(ctx => ctx.Response).Returns(response.Object);
    context.Setup(ctx => ctx.Session).Returns(session.Object);
    context.Setup(ctx => ctx.Server).Returns(server.Object);

    return context.Object;
}

但是我的测试一直失败,我得到:

CanRenderEmployeeListSystem.ApplicationException : No Http Context, 
                                                          No Session to Get!

这是要抛出的异常消息HttpContext.Current == null

我只需要 Session 对象“存在”,而不是存储在 Session 上的实际值。

你能告诉我我做错了什么吗?

谢谢。

4

2 回答 2

2

Session 属性中的 HttpContext.Current 不受您正在创建的模拟 HttpContextBase 的影响。也就是说,简单地创建一个本地 HttpContextBase 不会自动填充 HttpContext.Current。事实上 HttpContext 和 HttpContextBase 实际上并不相关。您必须使用 HttpContextWrapper 来统一它们。

因此,您最好将 HttpContextWrapper 实现传递给您的类 SessionVar。在下面的代码中,我将您的方法和属性更改为实例级别,以便我们可以在构造函数中设置上下文。如果没有将上下文传递给构造函数,我们假设为 HttpContext.Current,但您也可以将模拟实例传递给您的测试。

public class SessionVar
{
    HttpContextWrapper m_httpContext;

    public SessionVar(HttpContextWrapper httpContext = null)
    {
        m_httpContext = httpContext ?? new HttpContextWrapper(HttpContext.Current);
    }

    /// <summary>
    /// Gets the session.
    /// </summary>
    private HttpSessionState Session
    {
        get
        {
            if (m_httpContext == null)
                throw new ApplicationException("No Http Context, No Session to Get!");

            return m_httpContext.Session;
        }
    }

    public T Get<T>(string key)
    {
        return Session[key] == null ? default(T) : (T)Session[key];
    }
    ...
}
于 2012-10-13T00:30:01.030 回答
2

从长远来看,如果您为 SessionVar 类创建一个接口,您会更开心。在运行时使用您当前的实现(通过依赖注入)。在测试期间插入模拟。无需模拟所有这些 Http 运行时依赖项。

于 2012-10-12T22:59:40.487 回答