Подтвердить что ты не робот

Динамические атрибуты DataAnnotations

По-видимому, можно динамически присоединять атрибуты DataAnnotation к свойствам объекта во время выполнения и как таковые достичь динамической валидации.

Может ли кто-нибудь предоставить образец кода на этом?

4b9b3361

Ответ 1

MVC имеет крючок для предоставления собственного ModelValidatorProvider. По умолчанию MVC 2 использует подкласс класса ModelValidatorProvider, называемый DataAnnotationsModelValidatorProvider, который может использовать атрибуты System.DataAnnotations.ComponentModel.ValidationAttribute для проверки.

DataAnnotationsModelValidatorProvider использует отражение, чтобы найти все атрибуты ValidationAttributes и просто прокручивает коллекцию для проверки ваших моделей. Все, что вам нужно сделать, это переопределить метод GetValidators и ввести собственные атрибуты из того источника, который вы выберете. Я использую эту методику для проверки условных условий, свойства с атрибутом DataType.Email всегда передаются через регулярное выражение и используют этот метод для извлечения информации из базы данных, чтобы применить более ограничительные проверки для пользователей, не имеющих полномочий.

В следующем примере просто говорится: "Всегда делать какие-либо свойства FirstName":

 public class CustomMetadataValidationProvider : DataAnnotationsModelValidatorProvider
 {
    protected override IEnumerable<ModelValidator> GetValidators(ModelMetadata metadata, ControllerContext context, IEnumerable<Attribute> attributes)
    {
        //go to db if you want
        //var repository = ((MyBaseController) context.Controller).RepositorySomething;

        //find user if you need it
        var user = context.HttpContext.User;

        if (!string.IsNullOrWhiteSpace(metadata.PropertyName) && metadata.PropertyName == "FirstName")
            attributes = new List<Attribute>() {new RequiredAttribute()};

        return base.GetValidators(metadata, context, attributes);
    }
}

Все, что вам нужно сделать, это зарегистрировать поставщика в файле Global.asax.cs:

    protected void Application_Start()
    {
        ModelValidatorProviders.Providers.Add(new CustomMetadataValidationProvider());

        AreaRegistration.RegisterAllAreas();

        RegisterRoutes(RouteTable.Routes);
    }

Конечный результат:

end result

с этой моделью:

public class Person
{
    public string FirstName { get; set; }
    public string LastName { get; set; }
    public DateTime Birthday { get; set; }
}

Ответ 2

В вашем global.asax вам нужно очистить ModelValidatorProviders перед добавлением нового. В противном случае он добавит каждую аннотацию два раза, что даст вам "имена типа проверки в ненавязчивых правилах проверки клиента должны быть уникальными". - Ошибка.

protected void Application_Start()
{
    ModelValidatorProviders.Providers.Clear();
    ModelValidatorProviders.Providers.Add(new CustomMetadataValidationProvider());

    AreaRegistration.RegisterAllAreas();

    RegisterRoutes(RouteTable.Routes);
}

Ответ 3

Я не думаю, что вы можете добавлять атрибуты к членам во время выполнения, но вы, вероятно, можете использовать собственный поставщик метаданных для обработки этого для вас.

Вы должны проверить этот пост в блоге.

Ответ 4

Подход использования пользовательского MetadataValidationProvider с переопределенным GetValidators имеет несколько недостатков:

  • Некоторые атрибуты, такие как DisplayAttribute, не связаны с проверкой, поэтому добавление их на этапе проверки не работает.
  • Он не может быть надежным в будущем; обновление системы может привести к ее прекращению.

Если вы хотите, чтобы ваши динамически применяемые аннотации данных работали последовательно, вы можете подклассы DataAnnotationsModelMetadataProvider и DataAnnotationsModelValidatorProvider. После этого замените фреймворки через ModelMetadataProviders.Current и ModelValidatorProviders.Providers при запуске приложения. (Вы можете сделать это в Application_Start.)

При подклассе встроенных поставщиков систематический и, надеюсь, будущий способ применения ваших собственных атрибутов - переопределить GetTypeDescriptor. Я сделал это успешно, но это связано с созданием реализации ICustomTypeDescriptor и PropertyDescriptor, для чего потребовалось много кода и времени.