Skip to content

Commit

Permalink
Fixed detection of differing includes in set operands
Browse files Browse the repository at this point in the history
  • Loading branch information
roji committed Jun 27, 2019
1 parent b0d5dec commit 45d5732
Show file tree
Hide file tree
Showing 4 changed files with 39 additions and 25 deletions.
Original file line number Diff line number Diff line change
Expand Up @@ -560,7 +560,10 @@ static Expression UpdateEntityShaperEntityType(Expression shaperExpression, IEnt
case EntityShaperExpression entityShaperExpression:
return new EntityShaperExpression(newEntityType, entityShaperExpression.ValueBufferExpression, entityShaperExpression.Nullable);
case UnaryExpression unary when unary.NodeType == ExpressionType.Convert:
return Convert(UpdateEntityShaperEntityType(unary.Operand, newEntityType), unary.Type);
var newShaperExpression = UpdateEntityShaperEntityType(unary.Operand, newEntityType);
return newShaperExpression.Type == unary.Type
? newShaperExpression
: Convert(newShaperExpression, unary.Type);
case UnaryExpression unary when unary.NodeType == ExpressionType.ConvertChecked:
return ConvertChecked(UpdateEntityShaperEntityType(unary.Operand, newEntityType), unary.Type);
default:
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -874,33 +874,34 @@ private Expression ProcessSetOperation(MethodCallExpression methodCallExpression
var source2 = VisitSourceExpression(methodCallExpression.Arguments[1]);
var preProcessResult2 = PreProcessTerminatingOperation(source2);

// Compare the include chains from each side to make sure they're identical. We don't allow set operations over
// operands with different include chains.
var current1 = preProcessResult1.state.PendingIncludeChain?.NavigationTreeNode;
var current2 = preProcessResult2.state.PendingIncludeChain?.NavigationTreeNode;
while (true)
// Extract the includes from each side and compare to make sure they're identical.
// We don't allow set operations over operands with different includes.
var pendingIncludeFindingVisitor = new PendingIncludeFindingVisitor(false);
pendingIncludeFindingVisitor.Visit(preProcessResult1.state.PendingSelector.Body);
var pendingIncludes1 = pendingIncludeFindingVisitor.PendingIncludes;

pendingIncludeFindingVisitor = new PendingIncludeFindingVisitor(false);
pendingIncludeFindingVisitor.Visit(preProcessResult2.state.PendingSelector.Body);
var pendingIncludes2 = pendingIncludeFindingVisitor.PendingIncludes;

if (pendingIncludes1.Count != pendingIncludes2.Count)
{
if (current1 == null)
{
if (current2 == null)
{
break;
}
throw new NotSupportedException(CoreStrings.SetOperationWithDifferentIncludesInOperands);
}
throw new NotSupportedException(CoreStrings.SetOperationWithDifferentIncludesInOperands);
}

if (current2 == null)
foreach (var (i1, i2) in pendingIncludes1.Zip(pendingIncludes2, (i1, i2) => (i1, i2)))
{
if (i1.SourceMapping.RootEntityType != i2.SourceMapping.RootEntityType)
{
throw new NotSupportedException(CoreStrings.SetOperationWithDifferentIncludesInOperands);
}

if (current1.FromMappings.Zip(current2.FromMappings, (m1, m2) => (m1, m2))
.Any(t => !t.m1.SequenceEqual(t.m2)))
if (i1.NavTreeNode.Flatten()
.Zip(i2.NavTreeNode.Flatten(), (n1, n2) => (First: n1, Second: n2))
.Any(nodes => nodes.First.Navigation != nodes.Second.Navigation))
{
throw new NotSupportedException(CoreStrings.SetOperationWithDifferentIncludesInOperands);
}

(current1, current2) = (current1.Parent, current2.Parent);
}

// If the siblings are different types, one is derived from the other the set operation returns the less derived type.
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -139,8 +139,8 @@ protected override Expression VisitExtension(Expression extensionExpression)
result = NavigationExpansionHelpers.AddNavigationJoin(
result.source,
result.parameter,
pendingIncludeNode.Value,
pendingIncludeNode.Key,
pendingIncludeNode.SourceMapping,
pendingIncludeNode.NavTreeNode,
navigationExpansionExpression.State,
new List<INavigation>(),
include: true);
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -10,7 +10,15 @@ namespace Microsoft.EntityFrameworkCore.Query.NavigationExpansion.Visitors

public class PendingIncludeFindingVisitor : ExpressionVisitor
{
public virtual Dictionary<NavigationTreeNode, SourceMapping> PendingIncludes { get; } = new Dictionary<NavigationTreeNode, SourceMapping>();
private bool _skipCollectionNavigations;

public PendingIncludeFindingVisitor(bool skipCollectionNavigations = true)
{
_skipCollectionNavigations = skipCollectionNavigations;
}

public virtual List<(NavigationTreeNode NavTreeNode, SourceMapping SourceMapping)> PendingIncludes { get; } =
new List<(NavigationTreeNode, SourceMapping)>();

protected override Expression VisitMember(MemberExpression memberExpression) => memberExpression;
protected override Expression VisitInvocation(InvocationExpression invocationExpression) => invocationExpression;
Expand Down Expand Up @@ -69,14 +77,16 @@ protected override Expression VisitExtension(Expression extensionExpression)

private void FindPendingReferenceIncludes(NavigationTreeNode node, SourceMapping sourceMapping)
{
if (node.Navigation != null && node.Navigation.IsCollection())
if (_skipCollectionNavigations && node.Navigation != null && node.Navigation.IsCollection())
{
return;
}

if (node.Included == NavigationTreeNodeIncludeMode.ReferencePending && node.ExpansionMode != NavigationTreeNodeExpansionMode.ReferenceComplete)
if (node.ExpansionMode != NavigationTreeNodeExpansionMode.ReferenceComplete
&& (node.Included == NavigationTreeNodeIncludeMode.ReferencePending
|| !_skipCollectionNavigations && node.Included == NavigationTreeNodeIncludeMode.Collection))
{
PendingIncludes[node] = sourceMapping;
PendingIncludes.Add((node, sourceMapping));
}

foreach (var child in node.Children)
Expand Down

0 comments on commit 45d5732

Please sign in to comment.