属性级验证错误会阻碍类级验证的验证

时间:2010-06-23 06:36:07

标签: asp.net-mvc validation asp.net-mvc-2 data-annotations asp.net-mvc-3

Bounty获奖后更新

一个新的解决方案正在解决这个问题。请在此处参考ASP.NET MVC 3 Preview 1:http://weblogs.asp.net/scottgu/archive/2010/07/27/introducing-asp-net-mvc-3-preview-1.aspx

请参阅模型验证改进部分,在那里您将看到我的问题的解决方案。


原帖

参考我之前的帖子How to validate two properties with ASP.NET MVC 2,我在那里询问如何比较模型验证的两个属性。

我确实觉得答案很有用,但我遇到了一个完全不同的问题:

问题:如果属性级ValidationAttribute包含错误,则不验证类级别ValidationAttributes。

请考虑以下事项:

[EqualTo("Email", "EmailConfirm", ErrorMessage = "E-mailadresserne skal være ens")]
[EqualTo("Password", "PasswordConfirm", ErrorMessage = "Adgangskoderne skal være ens")]
[Bind(Exclude="UserId")]
public class EditSiteUser
{
    [Required(ErrorMessage="Du skal bekræfte adgangskode")]
    public string PasswordConfirm { get; set; }

    [Required(ErrorMessage="Du skal bekræfte e-mailadressen")]
    [Email(ErrorMessage="Ugyldig e-mailadresse")]
    public string EmailConfirm { get; set; }
    public int UserId { get; set; }

    [Required(ErrorMessage = "Du skal indtaste et brugernavn")]
    public string Username { get; set; }

    [Required(ErrorMessage = "Du skal indtaste en adgangskode")]
    public string Password { get; set; }

    [Required(ErrorMessage = "Du skal indtaste en e-mailadresse")]
    [Email(ErrorMessage = "Ugyldig e-mailadresse")]
    public string Email { get; set; }
}

这里我有两个类级别的属性,用于验证EmailConfirm和PasswordConfirm。

如果像Username这样的字段为空,从而产生错误,则永远不会验证两个EqualTo属性。

有没有人建议克服这个问题?

编辑:如果您需要有关此问题的更多信息,请在评论中提问,我将非常乐意为您提供所需的任何其他信息。

问题:

问:“如果属性级别检查失败,为什么对类级别检查进行验证很重要?”。

答:“因为这是表单的一部分,用户将信息输入到通过AJAX回发的表单中。当表单返回时,它应显示所有当前问题。”

问:“你在课堂上放置的EqualTo属性究竟是什么?它是一个自定义验证属性吗?如果是,它是如何工作的?它做了什么?”

答:EqualTo是一个类级ValidationAttribute,它比较类实例的两个属性的值。查找“PropertiesMustMatchAttribute”以获得类似的实现。

3 个答案:

答案 0 :(得分:6)

不支持此功能。如果任何属性级别验证失败,则不执行类级别验证。我建议你看看MVC Foolproof Validation。它扩展了MVC验证,以增加对或有属性验证的支持。我认为这可以解决这个特殊情况的问题。

项目网站声明它不适用于MVC2 RC,所以你必须下载源代码并让它自己运行/采纳他们的想法。

答案 1 :(得分:1)

要扩展到Scott Guthrie's article的链接,从MVC 3开始,您可以通过实现IValidatableObject接口来执行类级别验证。

这将在当前验证管道的上下文中工作,因此它可以嵌套在任意数量的自定义类和属性上,并继续返回完整的可能错误消息数组。

对于您的类,抛弃类级别属性,并添加一个名为Validate的方法,如下所示:

public class EditSiteUser : IValidatableObject
{
    public int UserId { get; set; }

    [Required(ErrorMessage = "Du skal indtaste et brugernavn")]
    public string Username { get; set; }

    [Required(ErrorMessage = "Du skal indtaste en adgangskode")]
    public string Password { get; set; }

    [Required(ErrorMessage="Du skal bekræfte adgangskode")]
    public string PasswordConfirm { get; set; }

    [Required(ErrorMessage = "Du skal indtaste en e-mailadresse")]
    [Email(ErrorMessage = "Ugyldig e-mailadresse")]
    public string Email { get; set; }

    [Required(ErrorMessage="Du skal bekræfte e-mailadressen")]
    [Email(ErrorMessage="Ugyldig e-mailadresse")]
    public string EmailConfirm { get; set; }


    public IEnumerable<ValidationResult> Validate(ValidationContext validationContext)
    { 
      // put whatever class level validation you want here

      if (Email !== EmailConfirm)
      {
          yield return new ValidationResult("E-mailadresserne skal være ens", new[] {"EmailConfirm"})
      }

      if (Password !== PasswordConfirm)
      {
          yield return new ValidationResult("Adgangskoderne skal være ens", new[] {"PasswordConfirm"})
      }
    }

}

您可以继续yield return尽可能多的验证邮件。

您可以使用@Html.ValidationSummary

在客户端上显示所有内容

如果您希望消息与特定控件一起显示,ValidationResult构造函数会对受影响属性的memberNames进行重载,并且您可以为此提供验证消息具有ValidationMessageFor HTML帮助程序的特定属性,如下所示:

@Html.ValidationMessageFor(Function(model) model.TestOne )

此外,值得一提的是,您可以使用CompareValidator轻松确保两个不同属性的值相等。使用此注释的好处是它会自动知道如何在客户端和服务器上强制执行此操作,而添加IValidatableObject只会在服务器上运行。

[DataType(DataType.Password)]
public string Password { get; set; }

[Compare("Password")]
[DataType(DataType.Password)]
public string ConfirmPassword { get; set; }

有关进一步阅读,Scott Guthrie的另一篇文章详细介绍了class level validation

答案 2 :(得分:0)

从您的示例中,您想要“确认其他输入框条目”框,正确的实现将是

 [EqualTo("Email", ErrorMessage = "E-mailadresserne skal være ens")]
 public string EmailConfirm { get; set; }

因为“错误”是对“确认”框的验证。 或换句话说, 您希望在确认框旁边显示错误消息,说明与电子邮箱中的错误消息不同。

这会将验证配置返回到属性,并解决您的问题。