KEMBAR78
[6.0.2] Put parentheses around IS NULL for non-bool operands by roji · Pull Request #26653 · dotnet/efcore · GitHub
Skip to content

Conversation

@roji
Copy link
Member

@roji roji commented Nov 12, 2021

Fixes #26652

Description

EF Core's SQL generation currently omits parentheses around SQL IS (NOT) NULL, in some cases where these are required.

Customer impact

When using database providers other than SQL Server, the LINQ expression Where(b => b.SomeBool = b.SomeNonBoolColumn.HasValue) returns incorrect results.

How found

Customer reported (originally in PomeloFoundation/Pomelo.EntityFrameworkCore.MySql#1309)

Regression

No

Testing

Added test for this scenario.

Risk

Very low, the fix simply adds parentheses around all occurrences of IS (NOT) NULL.

@roji
Copy link
Member Author

roji commented Nov 12, 2021

This would add lots of parentheses in various places, will hold off until @smitpatel takes a look, who's looking into parentheses questions elsewhere as well.

@smitpatel
Copy link
Contributor

@roji - I am not sure if I have a solution for this in 6.0.x. My fix requires adding new API and providers overriding it which is 7.0 work. How did this work in previous release? May be we can just mimic it.

@Isitar
Copy link

Isitar commented Nov 16, 2021

@smitpatel this worked before 6.x with the following code in npgsql:
npgsql/efcore.pg@313840d

@smitpatel
Copy link
Contributor

Sqlite has IS NULL as same priority as comparison but lower than arithmetic.
SqlServer doesn't really specify it explicitly but same as comparison.
Postgre has higher priority than comparison.

Given left-to-right behavior for same priority operations, we need to put brackets around IS NULL if it is being used during comparison operation. @roji - You should be able to put some hack in VisitSqlBinary method to make it work in patch release.

@roji
Copy link
Member Author

roji commented Nov 16, 2021

@smitpatel yeah, this isn't an urgent issue for me in Npgsql - I can bring back the hack I already had up to now.

But re Sqlite:

Sqlite has IS NULL as same priority as comparison but lower than arithmetic.

Right, but:

SELECT TRUE <> (NULL IS NULL); -- returns 0
SELECT TRUE <> NULL IS NULL; -- return 1

... because since the precedence is the same, SQLite evaluates left to right, giving incorrect results...

@roji roji force-pushed the IsNullParentheses branch from 93e99e2 to b72148b Compare November 17, 2021 22:36
@roji roji changed the base branch from main to release/6.0 November 17, 2021 22:36
@roji
Copy link
Member Author

roji commented Nov 17, 2021

Changed this to apply the same hack as for PostgreSQL - the advantage is that this is SQLite-only and does not change SQL Server's SQL. Rebased this on release/6.0 to consider for patching - the main branch should instead have the proper solution @smitpatel is working on (but keep the added test when merging this).

@roji roji force-pushed the IsNullParentheses branch 2 times, most recently from 455b94a to f5b3b5f Compare November 19, 2021 19:34
@roji roji requested a review from dougbu as a code owner November 19, 2021 19:34
@roji roji force-pushed the IsNullParentheses branch from f5b3b5f to d30cd2d Compare November 19, 2021 20:05
@roji roji force-pushed the IsNullParentheses branch from d30cd2d to b17740d Compare November 19, 2021 21:14
@roji roji changed the title Put parentheses around IS NULL for non-bool operands [release/6.0] Put parentheses around IS NULL for non-bool operands Jan 4, 2022
@roji
Copy link
Member Author

roji commented Jan 4, 2022

Yeah, @smitpatel I think this is only waiting for a review.

@wtgodbe
Copy link
Member

wtgodbe commented Jan 10, 2022

@roji looks like tactics didn't review this, if you want to get it in for February could you send mail to netcoretac before EOD?

@roji
Copy link
Member Author

roji commented Jan 10, 2022

Sent. If this is approved by tactics, could you (or someone else from the team) please merge it as it's quite late here in Europe?

@wtgodbe
Copy link
Member

wtgodbe commented Jan 10, 2022

Approved over email

@wtgodbe wtgodbe merged commit 021fbcb into dotnet:release/6.0 Jan 10, 2022
@roji roji deleted the IsNullParentheses branch January 11, 2022 00:00
@roji
Copy link
Member Author

roji commented Jan 11, 2022

Thanks @wtgodbe

@ajcvickers ajcvickers changed the title [release/6.0] Put parentheses around IS NULL for non-bool operands [6.0.2] Put parentheses around IS NULL for non-bool operands Jan 13, 2022
Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment

Projects

None yet

Development

Successfully merging this pull request may close these issues.

Parentheses put around IS NULL only if operand is of type boolean (6.0)

4 participants