Skip to content

Commit

Permalink
Fix to #17253 - Global Query Filter with reference to same entity
Browse files Browse the repository at this point in the history
When expanding defining query and/or query filters in nav rewrite we now replace instances of EntityQueryable of the entity for which defining query (or filter) is expanded into NavigationExpansionExpressions - this way we don't recursively try to visit and expand them.
  • Loading branch information
maumar committed Aug 21, 2019
1 parent 2bd8e78 commit d0b21ed
Show file tree
Hide file tree
Showing 3 changed files with 149 additions and 0 deletions.
Original file line number Diff line number Diff line change
Expand Up @@ -615,5 +615,33 @@ public override Expression Visit(Expression expression)
return base.Visit(expression);
}
}

private class SelfReferenceEntityQueryableRewritingExpressionVisitor : ExpressionVisitor
{
private readonly NavigationExpandingExpressionVisitor _navigationExpandingExpressionVisitor;
private readonly IEntityType _entityType;

public SelfReferenceEntityQueryableRewritingExpressionVisitor(
NavigationExpandingExpressionVisitor navigationExpandingExpressionVisitor,
IEntityType entityType)
{
_navigationExpandingExpressionVisitor = navigationExpandingExpressionVisitor;
_entityType = entityType;
}

protected override Expression VisitConstant(ConstantExpression constantExpression)
{
if (constantExpression.IsEntityQueryable())
{
var entityType = _navigationExpandingExpressionVisitor._queryCompilationContext.Model.FindEntityType(((IQueryable)constantExpression.Value).ElementType);
if (entityType == _entityType)
{
return _navigationExpandingExpressionVisitor.CreateNavigationExpansionExpression(constantExpression, entityType);
}
}

return base.VisitConstant(constantExpression);
}
}
}
}
Original file line number Diff line number Diff line change
Expand Up @@ -126,6 +126,8 @@ protected override Expression VisitConstant(ConstantExpression constantExpressio
{
var processedDefiningQueryBody = _parameterExtractingExpressionVisitor.ExtractParameters(definingQuery.Body);
processedDefiningQueryBody = _enumerableToQueryableMethodConvertingExpressionVisitor.Visit(processedDefiningQueryBody);
processedDefiningQueryBody = new SelfReferenceEntityQueryableRewritingExpressionVisitor(this, entityType).Visit(processedDefiningQueryBody);

navigationExpansionExpression = (NavigationExpansionExpression)Visit(processedDefiningQueryBody);

var expanded = ExpandAndReduce(navigationExpansionExpression, applyInclude: false);
Expand Down Expand Up @@ -170,6 +172,7 @@ private Expression ApplyQueryFilter(NavigationExpansionExpression navigationExpa
_parameterizedQueryFilterPredicateCache[rootEntityType] = filterPredicate;
}

filterPredicate = (LambdaExpression)new SelfReferenceEntityQueryableRewritingExpressionVisitor(this, entityType).Visit(filterPredicate);
var sequenceType = navigationExpansionExpression.Type.GetSequenceType();

// if we are constructing EntityQueryable of a derived type, we need to re-map filter predicate to the correct derived type
Expand Down
118 changes: 118 additions & 0 deletions test/EFCore.SqlServer.FunctionalTests/Query/QueryBugsTest.cs
Original file line number Diff line number Diff line change
Expand Up @@ -6212,6 +6212,124 @@ public PostDTO7983 From(Post7983 post)

#endregion

#region Bug17253

[ConditionalFact]
public virtual void Self_reference_in_query_filter_works()
{
using (CreateDatabase17253())
{
using (var context = new MyContext17253(_options))
{
var query = context.EntitiesWithQueryFilterSelfReference.Where(e => e.Name != "Foo");
var result = query.ToList();

AssertSql(
@"SELECT [e].[Id], [e].[Name]
FROM [EntitiesWithQueryFilterSelfReference] AS [e]
WHERE EXISTS (
SELECT 1
FROM [EntitiesWithQueryFilterSelfReference] AS [e0]) AND (([e].[Name] <> N'Foo') OR [e].[Name] IS NULL)");
}
}
}

[ConditionalFact]
public virtual void Self_reference_in_query_filter_works_when_nested()
{
using (CreateDatabase17253())
{
using (var context = new MyContext17253(_options))
{
var query = context.EntitiesReferencingEntityWithQueryFilterSelfReference.Where(e => e.Name != "Foo");
var result = query.ToList();

AssertSql(
@"SELECT [e].[Id], [e].[Name]
FROM [EntitiesReferencingEntityWithQueryFilterSelfReference] AS [e]
WHERE EXISTS (
SELECT 1
FROM [EntitiesWithQueryFilterSelfReference] AS [e0]
WHERE EXISTS (
SELECT 1
FROM [EntitiesWithQueryFilterSelfReference] AS [e1])) AND (([e].[Name] <> N'Foo') OR [e].[Name] IS NULL)");
}
}
}

public class MyContext17253 : DbContext
{
public DbSet<EntityWithQueryFilterSelfReference> EntitiesWithQueryFilterSelfReference { get; set; }
public DbSet<EntityReferencingEntityWithQueryFilterSelfReference> EntitiesReferencingEntityWithQueryFilterSelfReference { get; set; }

public DbSet<EntityWithQueryFilterCycle1> EntitiesWithQueryFilterCycle1 { get; set; }
public DbSet<EntityWithQueryFilterCycle2> EntitiesWithQueryFilterCycle2 { get; set; }
public DbSet<EntityWithQueryFilterCycle3> EntitiesWithQueryFilterCycle3 { get; set; }

public MyContext17253(DbContextOptions options) : base(options)
{
}

protected override void OnModelCreating(ModelBuilder modelBuilder)
{
modelBuilder.Entity<EntityWithQueryFilterSelfReference>().HasQueryFilter(e => EntitiesWithQueryFilterSelfReference.Any());
modelBuilder.Entity<EntityReferencingEntityWithQueryFilterSelfReference>().HasQueryFilter(e => Set<EntityWithQueryFilterSelfReference>().Any());

modelBuilder.Entity<EntityWithQueryFilterCycle1>().HasQueryFilter(e => EntitiesWithQueryFilterCycle2.Any());
modelBuilder.Entity<EntityWithQueryFilterCycle2>().HasQueryFilter(e => Set<EntityWithQueryFilterCycle3>().Any());
modelBuilder.Entity<EntityWithQueryFilterCycle3>().HasQueryFilter(e => EntitiesWithQueryFilterCycle1.Any());
}
}

private SqlServerTestStore CreateDatabase17253()
=> CreateTestStore(
() => new MyContext17253(_options),
context =>
{
context.EntitiesWithQueryFilterSelfReference.Add(new EntityWithQueryFilterSelfReference { Name = "EntityWithQueryFilterSelfReference" });
context.EntitiesReferencingEntityWithQueryFilterSelfReference.Add(new EntityReferencingEntityWithQueryFilterSelfReference { Name = "EntityReferencingEntityWithQueryFilterSelfReference" });
context.EntitiesWithQueryFilterCycle1.Add(new EntityWithQueryFilterCycle1 { Name = "EntityWithQueryFilterCycle1_1" });
context.EntitiesWithQueryFilterCycle2.Add(new EntityWithQueryFilterCycle2 { Name = "EntityWithQueryFilterCycle2_1" });
context.EntitiesWithQueryFilterCycle3.Add(new EntityWithQueryFilterCycle3 { Name = "EntityWithQueryFilterCycle3_1" });
context.SaveChanges();
ClearLog();
});

public class EntityWithQueryFilterSelfReference
{
public int Id { get; set; }
public string Name { get; set; }
}

public class EntityReferencingEntityWithQueryFilterSelfReference
{
public int Id { get; set; }
public string Name { get; set; }
}

public class EntityWithQueryFilterCycle1
{
public int Id { get; set; }
public string Name { get; set; }
}

public class EntityWithQueryFilterCycle2
{
public int Id { get; set; }
public string Name { get; set; }
}

public class EntityWithQueryFilterCycle3
{
public int Id { get; set; }
public string Name { get; set; }
}

#endregion

private DbContextOptions _options;

private SqlServerTestStore CreateTestStore<TContext>(
Expand Down

0 comments on commit d0b21ed

Please sign in to comment.