在Aspnet Core中加载MetaData

时间:2018-04-19 11:41:12

标签: c# asp.net-mvc entity-framework asp.net-core

我正在使用AspNet Core在MVC模板中构建一个项目。

我使用过Entity Framework,我已经搭建了现有的数据库。现在,我想在某个类中添加一些数据注释,但我不想编辑脚手架自动生成的类,所以我尝试使用元数据和覆盖现有方法saveChanges

Scaffolding自动生成的Users.cs

public partial class Users
{
    public int UserId { get; set; }
    ...
    // If I have [MaxLength(5, ErrorMessage = "Too short")] here, it works
    public string Email { get; set; }
}

UsersMetadata.cs (也试过了Users.Metadata.cs,否则没有改变)

[ModelMetadataType(typeof(UsersModelMetaData))]
public partial class Users { }

public class UsersModelMetaData
{
    [MaxLength(5, ErrorMessage = "Too short")]
    public string Email { get; set; }
}

MyContext:DbContext类

public override int SaveChanges()
{
    var entities = from e in ChangeTracker.Entries()
                   where e.State == EntityState.Added
                       || e.State == EntityState.Modified
                   select e.Entity;
    foreach (var entity in entities)
    {               
    var validationContext = new ValidationContext(entity);
        Validator.ValidateObject(entity, validationContext, validateAllProperties: true);
    }

    return base.SaveChanges();
}

所以,即使这似乎是正确的解决方案(我整个上午都在搜索),但它不起作用:问题似乎是UsersModelMetaData内的数据注释未被读取,因为如果我将数据注释直接放在Users.cs文件中,方法saveChanges()将抛出异常。

我确实找到了这个解决方案 - > https://stackoverflow.com/a/30127682/6070423< - 但它基于AspNet,使用AspNet Core我无法使用AssociatedMetadataTypeTypeDescriptionProvider

知道如何解决这个问题吗?

1 个答案:

答案 0 :(得分:0)

所以,在我问ASP.Net Core forum之后,感谢Edward Z回答,我确实找到了一种方法让它发挥作用。

基本上,我没有使用ModelMetadataType创建一个部分类,而是创建了一个包含所有需要验证的属性的类(因此,对于脚手架创建的每个类,我都会有一个这样的类)。这样的事情。

// Class generated by scaffolding
public partial class Users
    {
        public int UserId { get; set; }
        public string UserName { get; set; }
        public string Password { get; set; }
        public string Email { get; set; }
    }

// Class I did create. Notice that the name could be anything
//   but should contain at least "Users", which is the name of
//   the table generated by scaffolding.
// Also notice that isUnique is a Custom Attribute I've wrote.
public class UsersMetaData
    {
        [MinLength(2, ErrorMessage = " too short")]   
        public string UserName { get; set; }
        [MinLength(2, ErrorMessage = " too short psw")]
        public string Password { get; set; }
        [IsUnique(ErrorMessage = "Email already exists")]
        public string Email { get; set; }            
    }

之后,我创建了一个映射器,它将脚手架自动生成的类的实例映射到我所做的相对类。像这样:

public static object MapEntity(object entityInstance)
   {
        var typeEntity = entityInstance.GetType();
        var typeMetaDataEntity = Type.GetType(typeEntity.FullName + "MetaData");

        if (typeMetaDataEntity == null)
            throw new Exception();

        var metaDataEntityInstance = Activator.CreateInstance(typeMetaDataEntity);

        foreach (var property in typeMetaDataEntity.GetProperties())
        {
            if (typeEntity.GetProperty(property.Name) == null)
                throw new Exception();

            property.SetValue(
                metaDataEntityInstance,
                typeEntity.GetProperty(property.Name).GetValue(entityInstance));
        }

        return metaDataEntityInstance;
    }

请注意,在第一行中,从实例的类型(即Users)开始,我通过向该名称添加字符串“Metadata”来检索我创建的相应类。然后,我所做的只是创建一个UserMetaData类的实例,分配相同的Users实例值。

最后,当这个映射器被调用时?在SaveChanges()中,当我进行验证时。那是因为验证是在UserMetaData类上进行的。

public override int SaveChanges()
        {
            var entities = from e in ChangeTracker.Entries()
                           where e.State == EntityState.Added
                               || e.State == EntityState.Modified
                           select e.Entity;
            foreach (var entity in entities)
            {
                var metaDataEntityInstance = EntityMapper.MapEntity(entity);
                var validationContext = new ValidationContext(metaDataEntityInstance);
                Validator.ValidateObject(
                    metaDataEntityInstance,
                    validationContext,
                    validateAllProperties: true);
            }

            return base.SaveChanges();
        }

这有点棘手,但一切正常!