如何在 ASP.NET 中干净地共享数据模型、ViewModel 和 DTO 的自定义验证器 (ValidationAttribute)

How can I cleanly share a Custom Validator (ValidationAttribute) for a data model, a ViewModel, and a DTO in ASP.NET

我正在学习 ASP.NET MVC 课程。我正在使用 ASP.NET WebAPI 2 构建 REST Web API。该应用程序还包含标准 MVC 5 视图。我正在使用 DTO(数据传输对象)将 API 与数据模型分离。我制作了一个自定义的 ValidationAttribute,我已将其应用于我的数据模型中的 属性,并且我想对我的 DTO 上的 属性 和 属性 MVC 视图中使用的 ViewModel。

这需要将 ValidationContext.ObjectInstance 转换为正确的类型。我找到了一个简单的解决方案,但我觉得它不是很优雅,我想知道是否有更好的方法来做到这一点。

我说的具体ValidationAttribute和属性:

[Min18YearsIfAMember]
public DateTime? DateOfBirth { get; set; }

在解决方案的上下文中(为简洁起见删除了一些细节,包括 CustomerViewModel):

public class Customer
{
    public int Id { get; set; }
    public string Name { get; set; }
    public MembershipType MembershipType { get; set; }
    public byte MembershipTypeId { get; set; }

    [Min18YearsIfAMember]
    public DateTime? DateOfBirth { get; set; }
}    


public class CustomerDto
{
    public int Id { get; set; }
    public string Name { get; set; }
    public byte MembershipTypeId { get; set; }

    [Min18YearsIfAMember]
    public DateTime? DateOfBirth { get; set; }  
}

public class Min18YearsIfAMemberAttribute : ValidationAttribute
{
    protected override ValidationResult IsValid(object value, ValidationContext validationContext)
    {
        // Check it here
        var customer = validationContext.ObjectInstance as Customer;
        if (customer != null)
            return DoValidation(customer.MembershipTypeId, customer.DateOfBirth);

        // Check it here
        var customerVm = validationContext.ObjectInstance as CustomerViewModel;
        if (customerVm  != null)
            return DoValidation(customerVm.MembershipTypeId, customerVm.DateOfBirth);

        // Yes I should probably check it here too
        var customerDto = validationContext.ObjectInstance as CustomerDto;
            return DoValidation(customerDto.MembershipTypeId, customerDto.DateOfBirth);
    }

    private ValidationResult DoValidation( int membershipTypeId, DateTime? DateOfBirth)
    { 
        // Do the validation....
    }
}

它是可读的,但我发现必须像这样检查每个可能的情况很难看 ValidationContext.ObjectInstance as Customer

有没有更好的方法?

在数据注释属性中,您可以在附加属性时指定依赖项 属性,并使用它可以验证对象类型的 属性:

public class Min18YearsIfAMemberAttribute : ValidationAttribute
{
    private string _dependentProperty { get; set; }

    public Min18YearsIfAMemberAttribute(string dependentProperty)
    {
        this._dependentProperty = dependentProperty;
    }

    protected override ValidationResult IsValid(object value, ValidationContext validationContext)
    {
        var field = validationContext.ObjectType.GetProperty(_dependentProperty);
        if (field != null)
        {
            var dependentValue = (byte)field.GetValue(validationContext.ObjectInstance, null);
            
            return DoValidation(dependentValue, (DateTime?)value);
        }
        else
        {
            return new ValidationResult("<Your message here>");
        }
    }

    private ValidationResult DoValidation( int membershipTypeId, DateTime? DateOfBirth)
    { 
        // Do the validation....
    }

现在,在附加属性时,我们指定从属 属性 名称 [Min18YearsIfAMember("MembershipTypeId")

public class Customer
{
    public int Id { get; set; }
    public string Name { get; set; }
    public MembershipType MembershipType { get; set; }
    public byte MembershipTypeId { get; set; }

    [Min18YearsIfAMember(nameof(MembershipTypeId))]
    public DateTime? DateOfBirth { get; set; }
}