-
Notifications
You must be signed in to change notification settings - Fork 3.2k
New issue
Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.
By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.
Already on GitHub? Sign in to your account
Support projecting different navigations (but same type) via conditional operator #34589
Comments
Will it be fix ? |
Possibly related #34589. |
This working in ef6.. My project depends heavily on this |
@cincuranet #34589 is this issue 🤔 |
You are right. So it's not possibly related, but surely related. ;) But it doesn't matter. During a triage with @roji last week we concluded it (whatever it was) was not related. |
IIUC the issue is that This is (hopefully) going to be true for the condition (as it should always be a boolean value), but the true/false expressions might transport much more complex types (they might be navigations to other objects/collections/new{} objects, ...). I am unsure if this is 100% general, but distributing some of the operations applied to the conditional might help in many cases. In this case the transformation would be: |
Yeah, good points @ranma42. First, C# itself constrains the result of the conditional operation to be the same .NET type (e.g. you can't have
We should probably be concentrating on the 1st case here, possibly splitting the 2nd case out to another issue if needed.
Yeah, that's possible... Though for the general case we still must support the basic translation here; for example, the results of the conditional expression may be projected out without anything being composed on top. And once the general case is done, I'm not sure lowering the member access into the conditional is good (the SQL becomes more complex with duplication...). @dzyranov given this worked on EF6, could you please post the SQL you got there? Also, assuming your actual query above is a simplification ( |
Yes, the 2nd case would definitely be a further step.
I believe that this issue should essentially affect collection operations such as
I think that the opportunities for improving the SQL are unrelated to this. A slightly modified version of the original snippet (changed so that the conditional matches your example of using 2 different values in the branches): // @nuget: Microsoft.EntityFrameworkCore.Sqlite -Version 8.0.8
using System;
using System.Data;
using System.Linq;
using Microsoft.EntityFrameworkCore;
using var db = new BloggingContext();
db.Database.EnsureDeleted();
db.Database.EnsureCreated();
db.Blogs
.Where(x => (x.Id == 0 ? x.LeftChild.Id : x.RightChild.Id) != null)
.ToList();
db.Blogs
.Where(x => (x.Id == 0 ? x.LeftChild.Url : x.RightChild.Url) != null)
.ToList();
db.Blogs
.Where(x => (x.Id == 0 ? x.LeftChild : x.RightChild).Url != null)
.ToList();
public class BloggingContext : DbContext
{
public DbSet<Blog> Blogs { get; set; }
protected override void OnConfiguring(DbContextOptionsBuilder options)
=> options
.LogTo(Console.WriteLine, Microsoft.Extensions.Logging.LogLevel.Information)
.UseSqlite($"Data Source=test.db");
protected override void OnModelCreating(ModelBuilder modelBuilder)
{
modelBuilder.Entity<Blog>().HasOne(x => x.LeftChild);
modelBuilder.Entity<Blog>().HasOne(x => x.RightChild);
}
}
public class Blog
{
public int Id { get; set; }
public string Url { get; set; }
public virtual Blog? LeftChild { get; set; }
public virtual Blog? RightChild { get; set; }
} This produces a first query that is: SELECT "b"."Id", "b"."LeftChildId", "b"."RightChildId", "b"."Url"
FROM "Blogs" AS "b"
LEFT JOIN "Blogs" AS "b0" ON "b"."LeftChildId" = "b0"."Id"
LEFT JOIN "Blogs" AS "b1" ON "b"."RightChildId" = "b1"."Id"
WHERE CASE
WHEN "b"."Id" = 0 THEN "b0"."Id"
ELSE "b1"."Id"
END = 42 I agree that is inefficient because of the JOINs; ideally this should be SELECT "b"."Id", "b"."LeftChildId", "b"."RightChildId", "b"."Url"
FROM "Blogs" AS "b"
WHERE CASE
WHEN "b"."Id" = 0 THEN "b"."LeftChildId"
ELSE "b"."RightChildId"
END = 42 but AFAICT this is unrelated to the conditional (EFCore does not simplify FK accesses; this seems to be #28077 and #33067). The more relevant query is: SELECT "b"."Id", "b"."LeftChildId", "b"."RightChildId", "b"."Url"
FROM "Blogs" AS "b"
LEFT JOIN "Blogs" AS "b0" ON "b"."LeftChildId" = "b0"."Id"
LEFT JOIN "Blogs" AS "b1" ON "b"."RightChildId" = "b1"."Id"
WHERE CASE
WHEN "b"."Id" = 0 THEN "b0"."Url"
ELSE "b1"."Url"
END = 'foo' which looks mostly fine to me. An alternative query for this would be SELECT "b"."Id", "b"."LeftChildId", "b"."RightChildId", "b"."Url"
FROM "Blogs" AS "b"
WHERE CASE
WHEN "b"."Id" = 0 THEN (SELECT "b0"."Url" FROM "Blogs" AS "b0" WHERE "b"."LeftChildId" = "b0"."Id")
ELSE (SELECT "b1"."Url" FROM "Blogs" AS "b1" ON "b"."RightChildId" = "b1"."Id")
END = 'foo' but I believe this would actually be worse. I would definitely like to know what EF6 did in this case (and if @roji knows some trick to simplify the |
Use cases ef 6
ef core 8
Use
Sorry for my English |
@dzyranov we need the SQL. |
@ranma42 agree with everything you wrote...
I agree... In general JOIN syntax is preferable than scalar subqueries - database planners are usually able to do more with them... I think the ideal translation for your last query would be: SELECT "b"."Id", "b"."LeftChildId", "b"."RightChildId", "b"."Url"
FROM "Blogs" AS "b"
LEFT JOIN "Blogs" AS "b0" ON CASE WHEN "b"."Id" = 0 THEN "b"."LeftChildId" = "b0"."Id" ELSE "b"."RightChildId" = "b0"."Id" END
-- LEFT JOIN "Blogs" AS "b0" ON "b"."LeftChildId" = "b0"."Id"
-- LEFT JOIN "Blogs" AS "b1" ON "b"."RightChildId" = "b1"."Id"
WHERE [b0].[Url] = 'foo' That is, recognize that the join itself is conditional (but against the same table). Note #33745 (comment), where we're just now discussing allowing arbitrary predicates in join conditions. While that's easy enough to do, actuallyrecognizing the pattern and transforming it appropriately is likely to be quite non-trivial. |
|
i.e. for issue dotnet#34589
File a bug
Include stack traces
Include provider and version information
EF Core version: 8.08
Database provider: Microsoft.EntityFrameworkCore.SqlLite
Target framework: .NET 8.0
Operating system: Visual Studio 2022 17.4
The text was updated successfully, but these errors were encountered: