mirror of https://github.com/postgres/postgres
Consider a clause to be outerjoin_delayed if it references the nullable side
of any lower outer join, even if it also references the non-nullable side and so could not get pushed below the outer join anyway. We need this in case the clause is an OR clause: if it doesn't get marked outerjoin_delayed, create_or_index_quals() could pull an indexable restriction for the nullable side out of it, leading to wrong results as demonstrated by today's bug report from toruvinn. (See added regression test case for an example.) In principle this has been wrong for quite a while. In practice I don't think any branch before 8.3 can really show the failure, because create_or_index_quals() will only pull out indexable conditions, and before 8.3 those were always strict. So though we might have improperly generated null-extended rows in the outer join, they'd get discarded from the result anyway. The gating factor that makes the failure visible is that 8.3 considers "col IS NULL" to be indexable. Hence I'm not going to risk back-patching further than 8.3.
This commit is contained in:
parent
f6c1dece9a
commit
dcc2334736
|
@ -8,7 +8,7 @@
|
||||||
*
|
*
|
||||||
*
|
*
|
||||||
* IDENTIFICATION
|
* IDENTIFICATION
|
||||||
* $PostgreSQL: pgsql/src/backend/optimizer/plan/initsplan.c,v 1.139 2008/04/01 00:48:33 tgl Exp $
|
* $PostgreSQL: pgsql/src/backend/optimizer/plan/initsplan.c,v 1.140 2008/06/27 20:54:37 tgl Exp $
|
||||||
*
|
*
|
||||||
*-------------------------------------------------------------------------
|
*-------------------------------------------------------------------------
|
||||||
*/
|
*/
|
||||||
|
@ -1092,14 +1092,15 @@ check_outerjoin_delay(PlannerInfo *root, Relids *relids_p,
|
||||||
(ojinfo->is_full_join &&
|
(ojinfo->is_full_join &&
|
||||||
bms_overlap(relids, ojinfo->min_lefthand)))
|
bms_overlap(relids, ojinfo->min_lefthand)))
|
||||||
{
|
{
|
||||||
/* yes; have we included all its rels in relids? */
|
/* yes, so set the result flag */
|
||||||
|
outerjoin_delayed = true;
|
||||||
|
/* have we included all its rels in relids? */
|
||||||
if (!bms_is_subset(ojinfo->min_lefthand, relids) ||
|
if (!bms_is_subset(ojinfo->min_lefthand, relids) ||
|
||||||
!bms_is_subset(ojinfo->min_righthand, relids))
|
!bms_is_subset(ojinfo->min_righthand, relids))
|
||||||
{
|
{
|
||||||
/* no, so add them in */
|
/* no, so add them in */
|
||||||
relids = bms_add_members(relids, ojinfo->min_lefthand);
|
relids = bms_add_members(relids, ojinfo->min_lefthand);
|
||||||
relids = bms_add_members(relids, ojinfo->min_righthand);
|
relids = bms_add_members(relids, ojinfo->min_righthand);
|
||||||
outerjoin_delayed = true;
|
|
||||||
/* we'll need another iteration */
|
/* we'll need another iteration */
|
||||||
found_some = true;
|
found_some = true;
|
||||||
}
|
}
|
||||||
|
|
|
@ -2321,3 +2321,15 @@ where f2 = 53;
|
||||||
53 | | |
|
53 | | |
|
||||||
(1 row)
|
(1 row)
|
||||||
|
|
||||||
|
--
|
||||||
|
-- regression test for improper extraction of OR indexqual conditions
|
||||||
|
-- (as seen in early 8.3.x releases)
|
||||||
|
--
|
||||||
|
select a.unique2, a.ten, b.tenthous, b.unique2, b.hundred
|
||||||
|
from tenk1 a left join tenk1 b on a.unique2 = b.tenthous
|
||||||
|
where a.unique1 = 42 and
|
||||||
|
((b.unique2 is null and a.ten = 2) or b.hundred = 3);
|
||||||
|
unique2 | ten | tenthous | unique2 | hundred
|
||||||
|
---------+-----+----------+---------+---------
|
||||||
|
(0 rows)
|
||||||
|
|
||||||
|
|
|
@ -2321,3 +2321,15 @@ where f2 = 53;
|
||||||
53 | | |
|
53 | | |
|
||||||
(1 row)
|
(1 row)
|
||||||
|
|
||||||
|
--
|
||||||
|
-- regression test for improper extraction of OR indexqual conditions
|
||||||
|
-- (as seen in early 8.3.x releases)
|
||||||
|
--
|
||||||
|
select a.unique2, a.ten, b.tenthous, b.unique2, b.hundred
|
||||||
|
from tenk1 a left join tenk1 b on a.unique2 = b.tenthous
|
||||||
|
where a.unique1 = 42 and
|
||||||
|
((b.unique2 is null and a.ten = 2) or b.hundred = 3);
|
||||||
|
unique2 | ten | tenthous | unique2 | hundred
|
||||||
|
---------+-----+----------+---------+---------
|
||||||
|
(0 rows)
|
||||||
|
|
||||||
|
|
|
@ -485,3 +485,13 @@ select * from
|
||||||
zt2 left join zt3 on (f2 = f3)
|
zt2 left join zt3 on (f2 = f3)
|
||||||
left join zv1 on (f3 = f1)
|
left join zv1 on (f3 = f1)
|
||||||
where f2 = 53;
|
where f2 = 53;
|
||||||
|
|
||||||
|
--
|
||||||
|
-- regression test for improper extraction of OR indexqual conditions
|
||||||
|
-- (as seen in early 8.3.x releases)
|
||||||
|
--
|
||||||
|
|
||||||
|
select a.unique2, a.ten, b.tenthous, b.unique2, b.hundred
|
||||||
|
from tenk1 a left join tenk1 b on a.unique2 = b.tenthous
|
||||||
|
where a.unique1 = 42 and
|
||||||
|
((b.unique2 is null and a.ten = 2) or b.hundred = 3);
|
||||||
|
|
Loading…
Reference in New Issue