我正在尝试在 MVC 中的控制器上运行一个简单的测试。该控制器使用xVal和 DataAnnotations 验证输入。当我运行测试(通过 Resharper、NUnit 独立或 TestDriven.Net 使用 NUnit)时,它会使运行程序崩溃,并且没有像样的错误消息。在事件日志中,它只有一条相当通用的 .NET Runtime 2.0 错误报告消息,表明运行器是一个故障应用程序。
该错误是由对 ModelState.IsValid 的调用引起的(我知道这一点,因为当我把它拿出来时它运行良好)。此外,只有在我正常运行测试时才会发生崩溃。当我在调试模式下运行测试时,它工作正常。
当我删除对 xVal 的引用并使用 ModelState.AddModelError 在模型状态上设置错误时,它不会崩溃。
下面是被测控制器和测试类。你能看出这里有什么不对吗?
被测控制器
using System.Collections.Generic;
using System.ComponentModel;
using System.ComponentModel.DataAnnotations;
using System.Linq;
using System.Web.Mvc;
using xVal.ServerSide;
namespace TestModelState.Controllers
{
public class ThingController : Controller
{
[HttpPost]
public ActionResult Create(Thing thing)
{
try
{
var errors = DataAnnotationsValidationRunner.GetErrors(thing);
if (errors.Any())
{
throw new RulesException(errors);
}
}
catch (RulesException ex)
{
ex.AddModelStateErrors(ModelState, "thing");
}
if (ModelState.IsValid)
{
// Do some save stuff here
return RedirectToAction("Index");
}
else
{
return View();
}
}
}
public class Thing
{
[Required]
public string Name { get; set; }
}
internal static class DataAnnotationsValidationRunner
{
public static IEnumerable<ErrorInfo> GetErrors(object instance)
{
return from prop in TypeDescriptor.GetProperties(instance).Cast<PropertyDescriptor>()
from attribute in prop.Attributes.OfType<ValidationAttribute>()
where !attribute.IsValid(prop.GetValue(instance))
select new ErrorInfo(prop.Name, attribute.FormatErrorMessage(string.Empty), instance);
}
}
}
测试班
using System.Web.Mvc;
using NUnit.Framework;
using TestModelState.Controllers;
namespace TestModelState.Tests
{
[TestFixture]
public class ThingControllerTests
{
[Test]
public void Create_InvalidThing_SetsModelState()
{
// Arrange
var thingController = new ThingController();
var thing = new Thing();
// Act
var result = thingController.Create(thing);
// Assert
var viewResult = (ViewResult)result;
Assert.IsFalse(viewResult.ViewData.ModelState.IsValid);
}
}
}
版本 - ASP.Net MVC - 2.0.0.0,NUnit - 2.5.3.9345,xVal - 1.0.0.0
更新 当我改用以下语句(这是 ModelState.IsValid 在幕后所做的)时,不会发生崩溃......
var modelStateIsValid = ModelState.Values.All(ms => ms.Errors.Count == 0);
我仍然更喜欢使用 ModelState.IsValid,但至少这是一种解决方法。