Я прочитал некоторые ответы здесь: тестирование представлений и контроллеров и насмешливость, но я до сих пор не могу понять, как тестировать контроллер ASP.NET MVC, который читает и устанавливает значения Session (или любые другие основанные на контекстах переменные.) Как предоставить контекст (сеанс) для моих методов тестирования? Издевается над нами? У кого-нибудь есть примеры? В принципе, я хотел бы подделать сеанс, прежде чем я вызову метод контроллера, и контроллер будет использовать этот сеанс. Любые идеи?
ASP/NET MVC: Контроллеры с сеансами? Дразнящий?
Ответ 1
Отправляйте сообщение Стивена Вальтера в Faking the Controller Context:
Совет ASP.NET MVС# 12 - Подделка контекста контроллера
[TestMethod]
public void TestSessionState()
{
// Create controller
var controller = new HomeController();
// Create fake Controller Context
var sessionItems = new SessionStateItemCollection();
sessionItems["item1"] = "wow!";
controller.ControllerContext = new FakeControllerContext(controller, sessionItems);
var result = controller.TestSession() as ViewResult;
// Assert
Assert.AreEqual("wow!", result.ViewData["item1"]);
// Assert
Assert.AreEqual("cool!", controller.HttpContext.Session["item2"]);
}
Ответ 2
Структура ASP.NET MVC не очень дружелюбна (или, скорее, требует слишком много настроек для правильной подделки и вызывает слишком много трения при тестировании, IMHO) из-за использования абстрактных базовых классов вместо интерфейсов. Нам удавалось писать абстракции для каждого запроса и сеансового хранилища. Мы сохраняем эти абстракции очень легкими, а затем наши контроллеры зависят от этих абстракций для каждого запроса или для хранения в сеансе.
Например, здесь мы управляем файлами auth. У нас есть ISecurityContext:
public interface ISecurityContext
{
bool IsAuthenticated { get; }
IIdentity CurrentIdentity { get; }
IPrincipal CurrentUser { get; set; }
}
С конкретной реализацией, например:
public class SecurityContext : ISecurityContext
{
private readonly HttpContext _context;
public SecurityContext()
{
_context = HttpContext.Current;
}
public bool IsAuthenticated
{
get { return _context.Request.IsAuthenticated; }
}
public IIdentity CurrentIdentity
{
get { return _context.User.Identity; }
}
public IPrincipal CurrentUser
{
get { return _context.User; }
set { _context.User = value; }
}
}
Ответ 3
С MVC RC 1 ControllerContext обертывает HttpContext и предоставляет его как свойство. Это делает насмешливым намного легче. Чтобы высмеять переменную сеанса с помощью Moq, выполните следующие действия:
var controller = new HomeController();
var context = MockRepository.GenerateStub<ControllerContext>();
context.Expect(x => x.HttpContext.Session["MyKey"]).Return("MyValue");
controller.ControllerContext = context;
Подробнее см. сообщение Скотта Гуля.
Ответ 4
Я посмеялся довольно легко. Вот пример издевательства над httpContextbase (содержащий объекты запроса, сеанса и ответа) с помощью moq.
[TestMethod]
public void HowTo_CheckSession_With_TennisApp() {
var request = new Mock<HttpRequestBase>();
request.Expect(r => r.HttpMethod).Returns("GET");
var httpContext = new Mock<HttpContextBase>();
var session = new Mock<HttpSessionStateBase>();
httpContext.Expect(c => c.Request).Returns(request.Object);
httpContext.Expect(c => c.Session).Returns(session.Object);
session.Expect(c => c.Add("test", "something here"));
var playerController = new NewPlayerSignupController();
memberController.ControllerContext = new ControllerContext(new RequestContext(httpContext.Object, new RouteData()), playerController);
session.VerifyAll(); // function is trying to add the desired item to the session in the constructor
//TODO: Add Assertions
}
Надеюсь, что это поможет.
Ответ 5
У Скотта Ханзельмана есть сообщение о том, как создать загрузку файла quickapp с MVC и обсуждает обмен и конкретные адреса "Как издеваться над вещами, которые не являются издевательскими."
Ответ 6
Я использовал следующее решение - создание контроллера, на который наследуются все мои другие контроллеры.
public class TestableController : Controller
{
public new HttpSessionStateBase Session
{
get
{
if (session == null)
{
session = base.Session ?? new CustomSession();
}
return session;
}
}
private HttpSessionStateBase session;
public class CustomSession : HttpSessionStateBase
{
private readonly Dictionary<string, object> dictionary;
public CustomSession()
{
dictionary = new Dictionary<string, object>();
}
public override object this[string name]
{
get
{
if (dictionary.ContainsKey(name))
{
return dictionary[name];
} else
{
return null;
}
}
set
{
if (!dictionary.ContainsKey(name))
{
dictionary.Add(name, value);
}
else
{
dictionary[name] = value;
}
}
}
//TODO: implement other methods here as needed to forefil the needs of the Session object. the above implementation was fine for my needs.
}
}
Затем используйте код следующим образом:
public class MyController : TestableController { }
Ответ 7
Поскольку HttpContext статичен, я использую Typemock Isolator для его издевки, Typemock также имеет надстройку, созданную для модульное тестирование ASP.NET называется Ivonna.