// 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.AspNetCore.Mvc.ModelBinding.Validation; using Microsoft.Extensions.Localization; namespace Microsoft.AspNetCore.Mvc.DataAnnotations.Internal { /// /// Validates based on the given . /// public class DataAnnotationsModelValidator : IModelValidator { private static readonly object _emptyValidationContextInstance = new object(); 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.Name; var container = validationContext.Container; var context = new ValidationContext( instance: container ?? validationContext.Model ?? _emptyValidationContextInstance, serviceProvider: validationContext.ActionContext?.HttpContext?.RequestServices, items: null) { DisplayName = metadata.GetDisplayName(), MemberName = memberName }; var result = Attribute.GetValidationResult(validationContext.Model, context); if (result != ValidationResult.Success) { string errorMessage; if (_stringLocalizer != null && !string.IsNullOrEmpty(Attribute.ErrorMessage) && string.IsNullOrEmpty(Attribute.ErrorMessageResourceName) && Attribute.ErrorMessageResourceType == null) { errorMessage = GetErrorMessage(validationContext) ?? result.ErrorMessage; } else { errorMessage = result.ErrorMessage; } var validationResults = new List(); if (result.MemberNames != null) { foreach (var resultMemberName in result.MemberNames) { // ModelValidationResult.MemberName is used by invoking validators (such as ModelValidator) to // append construct the ModelKey for ModelStateDictionary. When validating at type level we // want 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 newMemberName = string.Equals(resultMemberName, memberName, StringComparison.Ordinal) ? null : resultMemberName; var validationResult = new ModelValidationResult(newMemberName, errorMessage); validationResults.Add(validationResult); } } if (validationResults.Count == 0) { // result.MemberNames was null or empty. validationResults.Add(new ModelValidationResult(memberName: null, message: errorMessage)); } return validationResults; } return Enumerable.Empty(); } private string GetErrorMessage(ModelValidationContextBase validationContext) { var adapter = _validationAttributeAdapterProvider.GetAttributeAdapter(Attribute, _stringLocalizer); return adapter?.GetErrorMessage(validationContext); } } }