ASP .Net MVC 3: 控制器行为单元测试

8

我对单元测试和Mock概念比较陌生。我正在尝试弄清楚如何为以下基本的开箱即用用户注册代码编写良好的测试用例:

[HttpPost]
public ActionResult Register(RegisterModel model)
{
    if (ModelState.IsValid)
    {
        // Attempt to register the user
        MembershipCreateStatus createStatus;
        Membership.CreateUser(model.UserName, model.Password, model.Email, null, null, true, null, out createStatus);

        if (createStatus == 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);
}

以下是我需要您的意见/帮助的一些具体问题:

  1. 我不一定想在ASP .Net成员资格数据库中创建一个新用户。
  2. 根据传入的模型,我如何确保用户成功注册或过程中是否存在错误。
2个回答

25

您的代码存在问题,您的操作依赖于静态方法:Membership.CreateUser。正如您所知,静态方法难以进行单元测试。

因此,您可以通过引入一个抽象层来降低耦合度:

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);

这是什么?这是一个静态方法调用。因此,我们需要像处理成员身份提供程序一样处理它,以减弱控制器和表单认证系统之间的耦合。


0

测试此方法您可以采用以下两种方式:

  1. 在测试类内创建一个新类,该类继承自 Membership 类并覆盖方法 CreateUser。
  2. 使用 Moq 对类进行模拟。

对于第一种情况,我将检查用户名是否等于“GoodUser”或“BadUser”,并生成一个 MembershipCreateStatus.Success 或其他状态。

对于第二种情况,我将设置两个方法,其思想与另一种方法相同。请参见此 link 示例。


网页内容由stack overflow 提供, 点击上面的
可以查看英文原文,
原文链接