// Copyright (c) .NET Foundation. All rights reserved. // Licensed under the Apache License, Version 2.0. See License.txt in the project root for license information. using System; using System.Collections.Generic; using System.ComponentModel.DataAnnotations; using System.Linq; using Microsoft.AspNet.Mvc.DataAnnotations; using Microsoft.Extensions.Localization; namespace Microsoft.AspNet.Mvc.ModelBinding.Validation { /// /// Validates based on the given . /// public class DataAnnotationsModelValidator : IModelValidator { private readonly IStringLocalizer _stringLocalizer; private readonly IValidationAttributeAdapterProvider _validationAttributeAdapterProvider; /// /// Create a new instance of . /// /// The that defines what we're validating. /// The used to create messages. /// The /// which 's will be created from. public DataAnnotationsModelValidator( IValidationAttributeAdapterProvider validationAttributeAdapterProvider, ValidationAttribute attribute, IStringLocalizer stringLocalizer) { if (validationAttributeAdapterProvider == null) { throw new ArgumentNullException(nameof(validationAttributeAdapterProvider)); } if (attribute == null) { throw new ArgumentNullException(nameof(attribute)); } _validationAttributeAdapterProvider = validationAttributeAdapterProvider; Attribute = attribute; _stringLocalizer = stringLocalizer; } /// /// The attribute being validated against. /// public ValidationAttribute Attribute { get; } /// /// Validates the context against the . /// /// The context being validated. /// An enumerable of the validation results. public IEnumerable Validate(ModelValidationContext validationContext) { if (validationContext == null) { throw new ArgumentNullException(nameof(validationContext)); } if (validationContext.ModelMetadata == null) { throw new ArgumentException( Resources.FormatPropertyOfTypeCannotBeNull( nameof(validationContext.ModelMetadata), typeof(ModelValidationContext)), nameof(validationContext)); } if (validationContext.MetadataProvider == null) { throw new ArgumentException( Resources.FormatPropertyOfTypeCannotBeNull( nameof(validationContext.MetadataProvider), typeof(ModelValidationContext)), nameof(validationContext)); } var metadata = validationContext.ModelMetadata; var memberName = metadata.PropertyName ?? metadata.ModelType.Name; var container = validationContext.Container; var context = new ValidationContext( instance: container ?? validationContext.Model, serviceProvider: validationContext.ActionContext?.HttpContext?.RequestServices, items: null) { DisplayName = metadata.GetDisplayName(), MemberName = memberName }; var result = Attribute.GetValidationResult(validationContext.Model, context); if (result != ValidationResult.Success) { // ModelValidationResult.MemberName is used by invoking validators (such as ModelValidator) to // construct the ModelKey for ModelStateDictionary. When validating at type level we want to append // the returned MemberNames if specified (e.g. person.Address.FirstName). For property validation, the // ModelKey can be constructed using the ModelMetadata and we should ignore MemberName (we don't want // (person.Name.Name). However the invoking validator does not have a way to distinguish between these // two cases. Consequently we'll only set MemberName if this validation returns a MemberName that is // different from the property being validated. var errorMemberName = result.MemberNames.FirstOrDefault(); if (string.Equals(errorMemberName, memberName, StringComparison.Ordinal)) { errorMemberName = null; } string errorMessage = null; if (_stringLocalizer != null && !string.IsNullOrEmpty(Attribute.ErrorMessage) && string.IsNullOrEmpty(Attribute.ErrorMessageResourceName) && Attribute.ErrorMessageResourceType == null) { errorMessage = GetErrorMessage(validationContext); } var validationResult = new ModelValidationResult(errorMemberName, errorMessage ?? result.ErrorMessage); return new ModelValidationResult[] { validationResult }; } return Enumerable.Empty(); } private string GetErrorMessage(ModelValidationContextBase validationContext) { var adapter = _validationAttributeAdapterProvider.GetAttributeAdapter(Attribute, _stringLocalizer); return adapter?.GetErrorMessage(validationContext); } } }