ASP.NET核心元数据类型属性不起作用

本文关键字:属性 不起作用 数据类型 NET 核心 ASP | 更新日期: 2023-09-27 18:28:44

我正在域模型类上使用MetaDataType属性。它应该将被引用类的属性信息移动到MetadataType属性已设置的类中。但它并不像宣传的那样。是什么导致了这里的问题?

[MetadataType(typeof(ComponentModelMetaData))]
public partial class Component
{
    public int Id { get; set; }
    public string Name { get; set; }
    public ICollection<Repo> Repos { get; set; }
    public string Description { get; set; }   
}

public class ComponentModelMetaData
{
    [Required(ErrorMessage = "Name is required.")]
    [StringLength(30, MinimumLength = 3, ErrorMessage = "Name length should be more than 3 symbols.")]
    public string Name { get; set; }
    public ICollection<Repo> Repos { get; set; }
    [Required(ErrorMessage = "Description is required.")]
    public string Description { get; set; }        
}

ASP.NET核心元数据类型属性不起作用

ASP.NET Core使用

Microsoft.AspNetCore.Mvc **ModelMetadataType** 

而不是

System.ComponentModel.DataAnnotations.**MetadataType** 

尝试将属性更改为[ModelMetadataType(typeof(ComponentModelMetaData))]

实体类:

namespace CoreProject.Persistence.EFCore
{
    public partial class User
    {
        public User()
        {
            Reader = new HashSet<Reader>();
            Writer = new HashSet<Writer>();
        }
        public int UserId { get; set; }
        public string Email { get; set; }
        public string Password { get; set; }
        public string PasswordHashKey { get; set; }
        public byte Role { get; set; }
        public string FirstName { get; set; }
        public string LastName { get; set; }
        public DateTime CreatedUtc { get; set; }
        public DateTime LastUpdateUtc { get; set; }
        public byte Status { get; set; }
        public bool Deleted { get; set; }
        public DateTime? ActivatedUtc { get; set; }
        public bool Test { get; set; }
        public virtual ICollection<Reader> Reader { get; set; }
        public virtual ICollection<Writer> Writer { get; set; }
    }
}

元数据(两者必须使用相同的命名空间):

namespace CoreProject.Persistence.EFCore
{
    [ModelMetadataType(typeof(IUserMetadata))]
    public partial class User : IUserMetadata
    {
        public string FullName => FirstName + " " + LastName;
    }
    public interface IUserMetadata
    {
        [JsonProperty(PropertyName = "Id")]
        int UserId { get; set; }
        [JsonIgnore]
        string Password { get; set; }
        [JsonIgnore]
        string PasswordHashKey { get; set; }
        [JsonIgnore]
        byte Role { get; set; }
    }
}

另一种方式。。。使用相同的命名空间

public class ApirolesMetaData
{
    [Required]
    public string Name { get; set; }
}

[ModelMetadataType(typeof(ApirolesMetaData))]
public partial class Apiroles
{
}

我也处于类似的情况,数据库在编码开始之前就已经存在。因此,DB首先似乎是一个自然的选择。生成的类,没有一个具有注释。所以添加了MetadataType,并找到了这个stackOverflow页面,所以尝试了ModelMetadataType。我对一个类和一个接口进行了尝试。

我尝试使用var validationContext = new ValidationContext(model);

而不是var editContext = new EditContext(model);

当我使用var result = Validator.TryValidateObject(model, validationContext, xxxxx, validateAllProperties: true);

当我尝试editContext.AddDataAnnotationsValidation(); 时,我以为我已经想通了

尽管这适用于带有注释的类,但(对我来说)对于使用[MetadataType][ModelMetadataType]属性的分部类不起作用。

我发现(对我来说)唯一可行的方法是创建一个新的类来包装你想要注释的模型。这意味着相同的属性,但getter和setter指向原始模型。但至少注释会起作用!

public class Model
{
    public int ID { get; set; }
    public string Name { get; set; }
}
public class AnnotatedModel
{
    private readonly Model model;
    public AnnotatedModel(Model model)
    {
        this.model = model;
    }
    [Range(1, int.MaxValue)]
    public int ID { get => model.ID; set => model.ID = value; }
    [Required]
    [StringLength(maximumLength: 10, ErrorMessage = "Name can't be longer than 10 characters.")]
    public string Name { get => model.Name; set => model.Name = value; }
}

因此,为了使用以上内容,我需要写下以下内容:

var model = new Model();
var annotatedModel = new AnnotatedModel(model);
var editContext = new EditContext(annotatedModel);
editContext.AddDataAnnotationsValidation();
var result = editContext.Validate();

如果有人能告诉我我缺了什么,那太好了!否则,我希望这项工作对某些人有用。