У вас проблема с вашим кодом. Ваше действие зависит от статического метода: Membership.CreateUser
. И, как вы знаете, статические методы являются PITA для модульного тестирования.
Чтобы вы могли ослабить связь, введя уровень абстракции:
public interface IMyService
{
MembershipCreateStatus CreateUser(string username, string password, string email);
}
, а затем есть некоторая реализация, которая будет использовать текущий поставщик членства:
public class MyService: IMyService
{
public MembershipCreateStatus CreateUser(string username, string password, string email)
{
MembershipCreateStatus status;
Membership.CreateUser(username, password, email, null, null, true, null, out status);
return status;
}
}
и, наконец, контроллер:
public class AccountController : Controller
{
private readonly IMyService _service;
public AccountController(IMyService service)
{
_service = service;
}
[HttpPost]
public ActionResult Register(RegisterModel model)
{
if (ModelState.IsValid)
{
// Attempt to register the user
var status = _service.CreateUser(model.UserName, model.Password, model.Email);
if (status == MembershipCreateStatus.Success)
{
FormsAuthentication.SetAuthCookie(model.UserName, false /* createPersistentCookie */);
return RedirectToAction("Index", "Home");
}
else
{
ModelState.AddModelError("", ErrorCodeToString(createStatus));
}
}
// If we got this far, something failed, redisplay form
return View(model);
}
}
Хорошо, теперь, когда мы ослабили связь, мы могли бы использовать фиктивную структуру, чтобы смоделировать сервис в модульном тесте и сделать его тривиальным.
Например, используя Rhino Mocks, вы можете создать следующие тесты, чтобы охватить 2 случая отказа:
[TestMethod]
public void Register_Action_Should_Redisplay_View_If_Model_Is_Invalid()
{
// arrange
var sut = new AccountController(null);
var model = new RegisterModel();
sut.ModelState.AddModelError("", "invalid email");
// act
var actual = sut.Register(model);
// assert
Assert.IsInstanceOfType(actual, typeof(ViewResult));
var viewResult = actual as ViewResult;
Assert.AreEqual(model, viewResult.Model);
}
[TestMethod]
public void Register_Action_Should_Redisplay_View_And_Add_Model_Error_If_Creation_Fails()
{
// arrange
var service = MockRepository.GenerateStub<IMyService>();
service
.Stub(x => x.CreateUser(null, null, null))
.IgnoreArguments()
.Return(MembershipCreateStatus.InvalidEmail);
var sut = new AccountController(service);
var model = new RegisterModel();
// act
var actual = sut.Register(model);
// assert
Assert.IsInstanceOfType(actual, typeof(ViewResult));
var viewResult = actual as ViewResult;
Assert.AreEqual(model, viewResult.Model);
Assert.IsFalse(sut.ModelState.IsValid);
}
Последний тест - это случай успеха. У нас все еще есть проблема с этим. Вопрос в следующей строке:
FormsAuthentication.SetAuthCookie(model.UserName, false);
Что это? Это статический вызов метода. Таким образом, мы поступаем так же, как и с провайдером членства, чтобы ослабить связь нашего контроллера и системы аутентификации форм.