1
0
mirror of https://github.com/postgres/postgres.git synced 2025-10-25 13:17:41 +03:00

Allow pushdown of HAVING clauses with grouping sets

In some cases, we may want to transfer a HAVING clause into WHERE in
hopes of eliminating tuples before aggregation instead of after.

Previously, we couldn't do this if there were any nonempty grouping
sets, because we didn't have a way to tell if the HAVING clause
referenced any columns that were nullable by the grouping sets, and
moving such a clause into WHERE could potentially change the results.

Now, with expressions marked nullable by grouping sets with the RT
index of the RTE_GROUP RTE, it is much easier to identify those
clauses that reference any nullable-by-grouping-sets columns: we just
need to check if the RT index of the RTE_GROUP RTE is present in the
clause.  For other HAVING clauses, they can be safely pushed down.

Author: Richard Guo
Discussion: https://postgr.es/m/CAMbWs4-NpzPgtKU=hgnvyn+J-GanxQCjrUi7piNzZ=upiCV=2Q@mail.gmail.com
This commit is contained in:
Richard Guo
2024-10-09 17:19:04 +09:00
parent 828e94c9d2
commit 67a54b9e83
3 changed files with 35 additions and 8 deletions

View File

@@ -1047,10 +1047,10 @@ subquery_planner(PlannerGlobal *glob, Query *parse, PlannerInfo *parent_root,
* cannot do so if the HAVING clause contains aggregates (obviously) or * cannot do so if the HAVING clause contains aggregates (obviously) or
* volatile functions (since a HAVING clause is supposed to be executed * volatile functions (since a HAVING clause is supposed to be executed
* only once per group). We also can't do this if there are any nonempty * only once per group). We also can't do this if there are any nonempty
* grouping sets; moving such a clause into WHERE would potentially change * grouping sets and the clause references any columns that are nullable
* the results, if any referenced column isn't present in all the grouping * by the grouping sets; moving such a clause into WHERE would potentially
* sets. (If there are only empty grouping sets, then the HAVING clause * change the results. (If there are only empty grouping sets, then the
* must be degenerate as discussed below.) * HAVING clause must be degenerate as discussed below.)
* *
* Also, it may be that the clause is so expensive to execute that we're * Also, it may be that the clause is so expensive to execute that we're
* better off doing it only once per group, despite the loss of * better off doing it only once per group, despite the loss of
@@ -1088,15 +1088,16 @@ subquery_planner(PlannerGlobal *glob, Query *parse, PlannerInfo *parent_root,
{ {
Node *havingclause = (Node *) lfirst(l); Node *havingclause = (Node *) lfirst(l);
if ((parse->groupClause && parse->groupingSets) || if (contain_agg_clause(havingclause) ||
contain_agg_clause(havingclause) ||
contain_volatile_functions(havingclause) || contain_volatile_functions(havingclause) ||
contain_subplans(havingclause)) contain_subplans(havingclause) ||
(parse->groupClause && parse->groupingSets &&
bms_is_member(root->group_rtindex, pull_varnos(root, havingclause))))
{ {
/* keep it in HAVING */ /* keep it in HAVING */
newHaving = lappend(newHaving, havingclause); newHaving = lappend(newHaving, havingclause);
} }
else if (parse->groupClause && !parse->groupingSets) else if (parse->groupClause)
{ {
Node *whereclause; Node *whereclause;

View File

@@ -860,6 +860,27 @@ explain (costs off)
-> Seq Scan on gstest2 -> Seq Scan on gstest2
(10 rows) (10 rows)
-- test pushdown of HAVING clause that does not reference any columns that are nullable by grouping sets
explain (costs off)
select a, b, count(*) from gstest2 group by grouping sets ((a, b), (a)) having a > 1 and b > 1;
QUERY PLAN
---------------------------------
GroupAggregate
Group Key: a, b
Group Key: a
Filter: (b > 1)
-> Sort
Sort Key: a, b
-> Seq Scan on gstest2
Filter: (a > 1)
(8 rows)
select a, b, count(*) from gstest2 group by grouping sets ((a, b), (a)) having a > 1 and b > 1;
a | b | count
---+---+-------
2 | 2 | 1
(1 row)
-- HAVING with GROUPING queries -- HAVING with GROUPING queries
select ten, grouping(ten) from onek select ten, grouping(ten) from onek
group by grouping sets(ten) having grouping(ten) >= 0 group by grouping sets(ten) having grouping(ten) >= 0

View File

@@ -279,6 +279,11 @@ explain (costs off)
select v.c, (select count(*) from gstest2 group by () having v.c) select v.c, (select count(*) from gstest2 group by () having v.c)
from (values (false),(true)) v(c) order by v.c; from (values (false),(true)) v(c) order by v.c;
-- test pushdown of HAVING clause that does not reference any columns that are nullable by grouping sets
explain (costs off)
select a, b, count(*) from gstest2 group by grouping sets ((a, b), (a)) having a > 1 and b > 1;
select a, b, count(*) from gstest2 group by grouping sets ((a, b), (a)) having a > 1 and b > 1;
-- HAVING with GROUPING queries -- HAVING with GROUPING queries
select ten, grouping(ten) from onek select ten, grouping(ten) from onek
group by grouping sets(ten) having grouping(ten) >= 0 group by grouping sets(ten) having grouping(ten) >= 0