-
-
Notifications
You must be signed in to change notification settings - Fork 3.1k
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
REGRESSION: Incorrect results when filtering on PostGIS/PostgreSQL or GeoPackage boolean columns #61072
Comments
This issue also effects Geopackages. There's also some probably-related weirdness going on as well. I've put together a simple test project; I'll shortly update the bug report with it and a clearer description of the issue and how to reproduce it. I looked at what I think is the file that would be handling this ( |
Test project / Narrowing down when the issue first appearedTLDR: The issues are observed to have appeared sometime between 3.35.0-Master 23f3af4 (2024-01-30) [exclusive] and 3.38.3-Grenoble Release 3.38 37f9e6e / 3.39.0-Master c50c923 (2024-09-13) [inclusive]. Test project and test dataThe following can be used to reproduce the test results:
Sample data for all tests (Postgres and Geopackage) has:
Both use a rule-based symbology with the rules set to the listed expressions. Both layers have Show Feature Count enabled . The items displayed on the map for each rule is shown; the following column (# in tree) contains the number shown in the layers tree for each rule category. [3.22.4 is a "known good" version. Later versions chosen to test all have changes to 3.22.4 (Ubuntu-maintained package)
Inconsistencies:
Oddities:
3.23.0-Master fcb8cf0 (2022-01-31)Same as 3.22.4 3.27.0-Master 4bfef0c (2022-08-29)Same as 3.22.4 3.35.0-Master 26798b8 (2024-01-30)Same as 3.22.4 3.35.0-Master 23f3af4 (2024-01-30)[One unrelated test failed during build but This version arguably has correct behavior. (It's my understanding that official support for
Selecting only 3.38.0-Master (?) 49d22d1 (2024-05-30)Not tested due to build errors for 3.38.3-Grenoble 37f9e6e (2024-09-13)a.k.a. Release 3.38
Inconsistencies:
Oddities for Postgres:
Oddities for Geopackage:
As mentioned above, it's my understanding that support for 3.39.0-Master c50c923 (2024-09-13)Same as 3.38.3-Grenoble 3.39.0-Master dd907d6 (2024-10-07)Same as 3.38.3-Grenoble 3.43.0-Master e634431 (2025-03-14)Same as 3.38.3-Grenoble |
Test results: SQL queriesI turned on Postgres logging and noted the queries used for both the correct and erroneous results. These are detailed below but the TLDR is that the queries used by both versions are identical in all cases. old: 3.35.0-Master 23f3af4 (2024-01-30) ["working"] Tests were run using the project and data mentioned in the previous comment. The headings indicate which rule or rules are active for each test. Old seen and new seen are what is displayed on the map with the numbers in parentheses being the feature counts shown for each rule (in order listed in heading). Note that the feature counts are determined at layer load time with values being computed only for active rules. Only the filter-specific portion of each query is shown as the preceding part of the queries (
|
old: | AND (("flag" IS NULL)) |
new: | AND (("flag" IS NULL)) |
expected: | null (1) |
old seen: | null (1) |
new seen: | null (1) |
"flag" IS NOT NULL
old: | AND (("flag" IS NOT NULL)) |
new: | AND (("flag" IS NOT NULL)) |
expected: | true & false (5) |
old seen: | true & false (5) |
new seen: | true & false (5) |
"flag" = false
old: | AND ("flag" IN (FALSE)) |
new: | AND ("flag" IN (FALSE)) |
expected: | false (2) |
old seen: | false (2) |
new seen: | false (2) |
"flag" != false
old: | AND (("flag" <> FALSE)) |
new: | AND (("flag" <> FALSE)) |
expected: | true (3) |
old seen: | true (3) |
new seen: | — (2) |
"flag" != false
and "flag" != true
old: | AND ((("flag" <> FALSE) OR ("flag" <> TRUE))) |
new: | AND ((("flag" <> FALSE) OR ("flag" <> TRUE))) |
expected: | true & false (3, 2) |
old seen: | true & false (3, 2) |
new seen: | true & false (2, 3) |
"flag" = true
old: | AND ("flag" IN (TRUE)) |
new: | AND ("flag" IN (TRUE)) |
expected: | true (3) |
old seen: | true (3) |
new seen: | true (3) |
"flag" != true
old: | AND (("flag" <> TRUE)) |
new: | AND (("flag" <> TRUE)) |
expected: | false (2) |
old seen: | false (2) |
new seen: | — (3) |
"flag" = 'false'
old: | AND ("flag" IN ('false')) |
new: | AND ("flag" IN ('false')) |
expected: | false (2) |
old seen: | false (2) |
new seen: | — (0) |
"flag" != false
and "flag" = 'false'
old: | AND ((("flag" <> FALSE) OR ("flag" = 'false'))) |
new: | AND ((("flag" <> FALSE) OR ("flag" = 'false'))) |
expected: | true & false (3, 2) |
old seen: | true & false (3, 2) |
new seen: | false (2, 0) |
"flag" != 'false'
old: | AND (("flag" <> 'false')) |
new: | AND (("flag" <> 'false')) |
expected: | true (3) |
old seen: | true (3) |
new seen: | — (0) |
"flag" != true
and "flag" != 'false'
old: | AND ((("flag" <> TRUE) OR ("flag" <> 'false'))) |
new: | AND ((("flag" <> TRUE) OR ("flag" <> 'false'))) |
expected: | true & false (2, 3) |
old seen: | true & false (2, 3) |
new seen: | true (3, 0) |
"flag" = 'true'
old: | AND ("flag" IN ('true')) |
new: | AND ("flag" IN ('true')) |
expected: | true (3) |
old seen: | true (3) |
new seen: | — (0) |
"flag" != true
and "flag" = 'true'
old: | AND ((("flag" <> TRUE) OR ("flag" = 'true'))) |
new: | AND ((("flag" <> TRUE) OR ("flag" = 'true'))) |
expected: | true & false (2, 3) |
old seen: | true & false (2, 3) |
new seen: | true (3, 0) |
"flag" != 'true'
old: | AND (("flag" <> 'true')) |
new: | AND (("flag" <> 'true')) |
expected: | false (2) |
old seen: | false (2) |
new seen: | — (0) |
"flag" != false
and "flag" != 'true'
old: | AND ((("flag" <> FALSE) OR ("flag" <> 'true'))) |
new: | AND ((("flag" <> FALSE) OR ("flag" <> 'true'))) |
expected: | true & false (3, 2) |
old seen: | true & false (3, 2) |
new seen: | false (2, 0) |
"flag" = 'f'
old: | AND ("flag" IN ('f')) |
new: | AND ("flag" IN ('f')) |
expected: | false (2) or — (0) |
old seen: | — (0) |
new seen: | — (0) |
"flag" != 'f'
old: | AND (("flag" <> 'f')) |
new: | AND (("flag" <> 'f')) |
expected: | true (3) or true & false (5) |
old seen: | true (5) |
new seen: | — (0) |
"flag" = 't'
old: | AND ("flag" IN ('t')) |
new: | AND ("flag" IN ('t')) |
expected: | true (3) or — (0) |
old seen: | — (0) |
new seen: | — (0) |
"flag" != 't'
old: | AND (("flag" <> 't')) |
new: | AND (("flag" <> 't')) |
expected: | false (2) or true & false (5) |
old seen: | false (5) |
new seen: | — (0) |
I've isolated the issue to a specific commit. Am evaluating things before commenting further. |
The issue was introduced in commit f1f6661e09b, merged in #56369. This commit was intended to require that both arguments to a binary boolean expression have boolean types. This has several issues:
|
I investigated the behavior of the interpretation of string and numeric values both by Qt (which QGIS relies on for such matters) and by a number of SQL databases typically used with QGIS [based on the list in documentation]. QtQGIS uses Qt's
SQLString conversions for all formats/databases is case-insensitive. PostgreSQL
GeoPackage
SQLite
Based on documentation, not tested: SQL Server
Implicit conversion to boolean is supported for column types Oracle (official support since release 23)
The official values above reflect the ad hoc methods used prior to official support being added. SAP HANA
There is no implicit conversion to boolean but values can be explicitly converted:
MySQL
MySQL does not have native support for boolean values; the constants |
…ng implicit value conversion) Fixes qgis#61072
What is the bug or the crash?
The handling of
boolean
fields for PostGIS/PostgreSQL and GeoPackage layers (and possibly others) broke some time between 3.35.0-Master 23f3af4 (2024-01-30) [working] and 3.38.3-Grenoble Release 3.38 37f9e6e / 3.39.0-Master c50c923 (2024-09-13) [both broken]. Filtering layer features based on the value of theboolean
column using a Rule-based Symbology results in incorrect feature filtering. Additional unexpected results and, at times, odd GUI behaviour are also seen when using such columns in a Provider Feature Filter (this additional behavior seems, at least in part, to be due to 3rd-party plugins).Erroneous behavior for Rule-based symbology is present for both my normal user profile and a new user profile. The erroneous/odd behavior for Provider Feature Filter is only present for my normal user profile (and not for a new profile). In all cases the behaviour on the older QGIS version is correct.
Most of what's below are test results. Click here to skip to the resolution and related commentary.
Update: Testing shows that the SQL queries are identical in both the correct and erroneous cases. This implies that the issue lies in how QGIS is interpreting or presenting the returned features.
[See also my comment below where I narrow down when the breaking change occurred.]
Here are the results of filtering a layer backed by a PostGIS table with 166914 rows. The data type of the
is_hidden
column is native PostgreSQLboolean
and the column contains a mix ofNULL
,true
, andfalse
values. The layer is added to the project and then either a Provider Feature Filter is applied to the layer or a Rule-based Symbology is used.The numbers shown below are, as appropriate:
Unexpected/Erroneous values/behavior shown in bold. Where behavior is different between my normal user profile and a new profile the former is shown in normal text and the latter is shown in italics.
The expressions shown below are specified in the direct SQL query, as the expression for the layer's Provider Feature Filter, or as the classification rule applied in the Rule-based symbology.
is_hidden
is the name of theboolean
column."is_hidden" IS NULL
47
47
"is_hidden" IS NOT NULL
"is_hidden" = 'true'
251
"is_hidden" != 'true'
"is_hidden" = 'false'
"is_hidden" != 'false'
251
"is_hidden" = true
251
"is_hidden" != true
"is_hidden" = false
"is_hidden" != false
251
1 No filter is shown as being applied (a blank filter was specified).
[This is reasonable behaviour.]
2 Error for old is "Filter expression parsing error: syntax error, unexpected end of file".
[This is reasonable behaviour.]
3 Error for current is "Filter expression parsing error: Incomplete expression. You might not have finished the full expression."
[This is reasonable behaviour.]
4 Value is initially 47 but changes to 35608.
Clicking OK results in the Filter window immediately re-opening; clicking OK again closes the window.
5 I have no idea where the 35608 value is coming from.
6 For my normal user profile (but not for a new profile) clicking OK results in the Filter window immediately re-opening; clicking OK again closes the window.
SQL = Result of running
select count(*) from "my_feature_table" where …
directly against the database.PT = Feature count shown when clicking Test when editing the Provider Feature Filter.
P = Feature count shown for layer in Layers tree when expression is set as the Provider Feature Filter.
RT = Feature count shown when clicking Test when editing the Rule-based Symbology rule.
R = Feature count shown for classification within layer in Layers tree when Rule-based Symbology rule is applied.
Old =
3.22.4-Białowieża
Cur. =
3.43.0-Master
(8ea86e8)Further investigation with a different table shows the following:
test
tree
map
table
test
tree
map
table
"is_edited"=true
"is_edited"!=false
"is_edited"=false
"is_edited"!=true
"is_edited"='true'
"is_edited"!='false'
"is_edited"='false'
"is_edited"!='true'
is_edited
is theboolean
column and the erroneous behavior is marked in bold.Old → 3.22.4-Białowieża
Cur. → 3.43.0-Master (8ea86e8)
SQL → Result of
select count(*) from "table_name" where
expression
query run directly against the database.Test → Value shown when clicking Test when editing the expression as a Rule-based symbology rule.
Tree →Number of features listed for category in Layers tree when expression is applied as a Rule-based symbology rule.
Map → Whether features are displayed on the map when expression is applied as a Rule-based symbology rule.
Table → Returned row count when expression is applied as an Advanced Filter (Expression) in the layer's Attribute Table window
Steps to reproduce the issue
geom geometry(MultiPolygonZ,4326)
and also containing aboolean
column.boolean
column values toNULL
, some totrue
(t
in PostgreSQL parlance), and some tofalse
(f
).Versions
PostgreSQL
9.5.2 (64-bit), Ubuntu-supplied package.
QGIS (old version, from Ubuntu-supplied Jammy package)
Differences are bolded.
QGIS (current Git
master
version, built from source)Differences are bolded.
Active Python plugins
The same plugins (from the same user profile directory) are used for both versions of QGIS.
Normal user profile
(normal user profile)
(new profile)
Supported QGIS version
New profile
Additional context
No response
The text was updated successfully, but these errors were encountered: