使用FluentValidation时,将DataAnnotation添加到类中

时间:2011-04-27 21:31:01

标签: asp.net-mvc data-annotations fluentvalidation

我使用FluentValidation框架在MVC项目中为我的模型添加验证和注释。

我需要将数据注释添加到模型的类级别。即,模型需要添加DisplayColumn属性。但是,由于我使用FluentValidation(并且将应用程序的ModelMetadataProvider设置为使用FluentValidation),即使我将DisplayColumn属性放在模型类上,也不会使用它。但是,我找不到使用FluentValidation添加该注释的方法。

有没有人知道如何让它发挥作用?

由于

1 个答案:

答案 0 :(得分:4)

我实际上不建议使用FluentValidationModelMetadataProvider - 这实际上只是一个实验性的添加(很可能会从下一个版本中删除),并且它不支持任何类级别的DataAnnotations(例如DisplayColumn)。我建议您仅使用FluentValidation进行验证,但坚持使用元数据属性。

话虽如此,如果你真的想让它工作,那么你可以通过使用仅用于元数据的自定义无操作验证器来实现:

public static class MetadataExt {
    public static IRuleBuilderOptions<T, TProperty> DisplayColumn<T, TProperty>(this IRuleBuilder<T, TProperty> rule) {
        var ruleBuilder = (FluentValidation.Internal.RuleBuilder<T, TProperty>)rule;
        ruleBuilder.Rule.AddValidator(new DisplayColumnWrapper(ruleBuilder.Rule.PropertyName));
        return ruleBuilder;
    }

    public class DisplayColumnWrapper : NoopPropertyValidator, IAttributeMetadataValidator {
        private string name;

        public DisplayColumnWrapper(string name) {
            this.name = name;
        }

        public override IEnumerable<ValidationFailure> Validate(PropertyValidatorContext context) {
            return Enumerable.Empty<ValidationFailure>();
        }

        public Attribute ToAttribute() {
            return new DisplayColumnAttribute(name);
        }
    }
}

......你可以这样使用:

public class Validator : AbstractValidator<SomeModel> {
    public Validator() {
        RuleFor(x => x.DisplayColumnProperty)
            .DisplayColumn();

    }
}

然后,您需要创建一个知道如何处理此问题的自定义ModelMetadataProvider:

public class ExtendedFVModelMetadataProvider : FluentValidationModelMetadataProvider {
    IValidatorFactory _validatorFactory;

    public ExtendedFVModelMetadataProvider(IValidatorFactory validatorFactory)
        : base(validatorFactory) {
        this._validatorFactory = validatorFactory;
    }

    public override ModelMetadata GetMetadataForType(Func<object> modelAccessor, Type modelType) {
        var validator = _validatorFactory.GetValidator(modelType);

        if (validator == null) {
            return base.GetMetadataForType(modelAccessor, modelType);
        }

        // Only look for the DisplayColumnWrapper 
        // There is a mismatch as MVC expects this to be defined at class-level, but FV defines everything at the property level.
        var displayColumns = from memberWithValidator in validator.CreateDescriptor().GetMembersWithValidators()
                             from propertyValidator in memberWithValidator
                             let wrapper = propertyValidator as MetadataExt.DisplayColumnWrapper
                             where wrapper != null
                             select wrapper.ToAttribute();

        var displayColumn = displayColumns.FirstOrDefault();

        // we found a displaycolumn, so pass it over to MVC to build the metadata.
        if (displayColumn != null) {
            return CreateMetadata(new[] { displayColumn }, null /* containerType */, modelAccessor, modelType, null /* propertyName */);
        }

        return base.GetMetadataForType(modelAccessor, modelType);

    }
}

提供程序重写GetMetadataForModel方法并查找使用DisplayColumn的任何属性。如果您还想支持任何其他自定义元数据扩展,则可能会扩展这一点。然后,您可以使用此提供程序代替FluentValidation附带的元数据提供程序。

但是,我仍然不推荐这种方法......该库旨在执行验证,而不是用于生成UI元数据。