过滤所有查询(尝试实现软删除)

Tom*_* C. 6 c# entity-framework entity-framework-core .net-core

我试图在EF Core 2.0中开展软删除行为.

public interface ISoftDeleteModel
{
    bool IsDeleted { get; set; }
}
Run Code Online (Sandbox Code Playgroud)

创建正确的列和软删除工作正常,但DbContext中的过滤实体不是.

我想在上下文中使用查询过滤,但我被卡住了.

protected override void OnModelCreating(ModelBuilder modelBuilder)
{
    Type entityType;
    // ^^^ it contains type of entity, eg. Blog, Post, etc. using
    // modelBuilder.Model.GetEntityTypes().First().Name and converting to Type

    var entity = modelBuilder.Entity(entityType);
    if(entityType.GetInterface("ISoftDeleteModel") != null)
    {
        // ??? how to access IsDeleted property ???
        entity.HasQueryFilter(x => !x.IsDeleted);
    }
}
Run Code Online (Sandbox Code Playgroud)

问题很简单 - 如何访问IsDeleted属性?

如果我知道实体的类型,例如.Post和Post实现的ISoftDeleteModel我可以这样做:

protected override void OnModelCreating(ModelBuilder modelBuilder)
{
    modelBuilder.Entity<Post>().HasQueryFilter(x => !x.IsDeleted);
}
Run Code Online (Sandbox Code Playgroud)

但我不知道这种类型.我试图实现简单的事情 - 所有实现此接口的模型都将被自动过滤.我错过了什么吗?

Iva*_*oev 18

无法测试确切的API,但一般方法是创建约束泛型方法并通过反射调用它:

public static class EFFilterExtensions
{
    public static void SetSoftDeleteFilter(this ModelBuilder modelBuilder, Type entityType)
    {
        SetSoftDeleteFilterMethod.MakeGenericMethod(entityType)
            .Invoke(null, new object[] { modelBuilder });
    }

    static readonly MethodInfo SetSoftDeleteFilterMethod = typeof(EFFilterExtensions)
               .GetMethods(BindingFlags.Public | BindingFlags.Static)
               .Single(t => t.IsGenericMethod && t.Name == "SetSoftDeleteFilter");

    public static void SetSoftDeleteFilter<TEntity>(this ModelBuilder modelBuilder) 
        where TEntity : class, ISoftDeleteModel
    {
        modelBuilder.Entity<TEntity>().HasQueryFilter(x => !x.IsDeleted);
    }
}
Run Code Online (Sandbox Code Playgroud)

现在你可以在你的内部使用这样的东西OnModelCreating:

foreach (var type in modelBuilder.Model.GetEntityTypes())
{
    if (typeof(ISoftDeleteModel).IsAssignableFrom(type.ClrType))
        modelBuilder.SetSoftDeleteFilter(type.ClrType);
}
Run Code Online (Sandbox Code Playgroud)

  • 请注意,当您对同一表/实体类型多次调用“HasQueryFilter”时,只有最新的调用才会生效。如果您需要链接多个查询过滤器,那么您可以使用此处的代码片段:https://github.com/dotnet/efcore/issues/10275#issuecomment-457504348 (2认同)