ASP.NET MVC Controller post方法单元测试:ModelState.IsValid始终为true

时间:2014-03-21 14:41:12

标签: c# asp.net-mvc unit-testing modelstate

我已经为ASP.NET MVC Web应用程序编写了第一个单元测试。一切正常,它给了我有价值的信息,但我不能在视图模型中测试错误。 ModelState.IsValid始终为true,即使未填充某些值(空字符串或null)。

我已经读过,当发布的数据映射到模型时,模型验证会发生,您需要编写一些代码来自行进行模型验证:

我已尝试过链接网页中提供的三个示例,但似乎对我不起作用。

一些代码:

我的观看模式

...
[Required(ErrorMessageResourceName = "ErrorFirstName", ErrorMessageResourceType = typeof(Mui))]
[MaxLength(50)]
[Display(Name = "Firstname", ResourceType = typeof(Mui))]
public string FirstName { get; set; }
...

控制器

...
 [HttpPost]
    public ActionResult Index(POSViewModel model)
    {
        Contract contract = contractService.GetContract(model.ContractGuid.Value);

        if (!contract.IsDirectDebit.ToSafe())
        {
            ModelState.Remove("BankName");
            ModelState.Remove("BankAddress");
            ModelState.Remove("BankZip");
            ModelState.Remove("BankCity");
            ModelState.Remove("AccountNr");
        }

        if (ModelState.IsValid)
        {
            ...

            contractValidationService.Create(contractValidation);
            unitOfWork.SaveChanges();

            return RedirectToAction("index","thanks");
        }
        else
        {
            return Index(model.ContractGuid.ToString());
        }
    }

我的单元测试

  posViewModel.FirstName = null;
  posViewModel.LastName = "";
 ...
 var modelBinder = new ModelBindingContext()
        {
            ModelMetadata = ModelMetadataProviders.Current.GetMetadataForType(() => posViewModel, posViewModel.GetType()),
            ValueProvider = new NameValueCollectionValueProvider(new System.Collections.Specialized.NameValueCollection(), CultureInfo.InvariantCulture)
        };
        var binder = new DefaultModelBinder().BindModel(new ControllerContext(), modelBinder);
        posController.ModelState.Clear();
        posController.ModelState.Merge(modelBinder.ModelState);

        ActionResult result = posController.Index(posViewModel);

        //Assert
        mockContractValidationService.Verify(m => m.Create(It.IsAny<ContractValidation>()), Times.Never);
        Assert.IsInstanceOfType(result, typeof(ViewResult));

在视图中,我使用的是不引人注目的JavaScript验证,它可以正常工作。

2 个答案:

答案 0 :(得分:81)

你试图同时测试两种不同的东西。控制器不能用于验证模型状态,只是根据验证结果的不同行为。因此,对控制器的单元测试不应该尝试测试验证,这应该在不同的测试中完成。在我看来,你应该有三个单元测试:

  1. 验证模型验证是否正确的方法
  2. 验证控制器在modelstate有效时是否正常运行的方法
  3. 验证控制器在modelstate无效时是否正常运行的方法
  4. 以下是如何做到的:

    1.模型验证

    [Test]
    public void test_validation()
    {
        var sut = new POSViewModel();
        // Set some properties here
        var context = new ValidationContext(sut, null, null);
        var results = new List<ValidationResult>();
        var isModelStateValid =Validator.TryValidateObject(sut, context, results, true);
    
        // Assert here
    }
    

    2.具有无效模型状态的控制器

    [Test]
    public void test_controller_with_model_error()
    {
        var controller = new PosController();
        controller.ModelState.AddModelError("test", "test");
    
        ActionResult result = posController.Index(new PosViewModel());
    
        // Assert that the controller executed the right actions when the model is invalid
    }
    

    3.具有有效模型状态的控制器

    [Test]
    public void test_controller_with_valid_model()
    {
        var controller = new PosController();
        controller.ModelState.Clear();
    
        ActionResult result = posController.Index(new PosViewModel());
    
        // Assert that the controller executed the right actions when the model is valid
    }
    

答案 1 :(得分:9)

我找到了这个解决方案:SO: Validation does not work when I use Validator.TryValidateObject结合解决方案@Kenneth提供:

[TestMethod]
    public void test_validation()
    {
        var sut = new POSViewModel();
        // Set some properties here
        var context = new ValidationContext(sut, null, null);
        var results = new List<ValidationResult>();
        TypeDescriptor.AddProviderTransparent(new AssociatedMetadataTypeTypeDescriptionProvider(typeof(POSViewModel), typeof(POSViewModel)), typeof(POSViewModel));

        var isModelStateValid = Validator.TryValidateObject(sut, context, results, true);

        // Assert here
    }

如果您有一个包含所有资源的类库,请不要忘记在测试项目中引用它。