我正在使用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
。
知道如何解决这个问题吗?
答案 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();
}
这有点棘手,但一切正常!