]> granicus.if.org Git - postgresql/commitdiff
Don't make FK-based selectivity estimates in inheritance situations.
authorTom Lane <tgl@sss.pgh.pa.us>
Wed, 2 Nov 2016 19:50:15 +0000 (15:50 -0400)
committerTom Lane <tgl@sss.pgh.pa.us>
Wed, 2 Nov 2016 19:50:21 +0000 (15:50 -0400)
The foreign-key-aware logic for estimation of join sizes (added in commit
100340e2d) blindly tried to apply the concept to rels that are actually
parents of inheritance trees.  This is just plain wrong so far as the
referenced relation is concerned, since the inheritance scan may well
produce lots of rows that are not participating in the constraint.  It's
wrong for the referencing relation too, for the same reason; although on
that end we could conceivably detect whether all members of the inheritance
tree have equivalent FK constraints pointing to the same referenced rel,
and then proceed more or less as we do now.  But pending somebody writing
code to do that, we must disable this, because it's producing completely
silly estimates when there's an FK linking the heads of inheritance trees.

Per bug #14404 from Clinton Adams.  Back-patch to 9.6 where the new
estimation logic came in.

Report: <20161028200412.15987.96482@wrigleys.postgresql.org>

src/backend/optimizer/util/plancat.c

index 5d18206b91b3527e1e9f8741545b1fefea6cfde0..ad07baa60c9ed0f6ec4a8b41c525ccbdf3455b0a 100644 (file)
@@ -53,7 +53,7 @@ get_relation_info_hook_type get_relation_info_hook = NULL;
 
 
 static void get_relation_foreign_keys(PlannerInfo *root, RelOptInfo *rel,
-                                                 Relation relation);
+                                                 Relation relation, bool inhparent);
 static bool infer_collation_opclass_match(InferenceElem *elem, Relation idxRel,
                                                          List *idxExprs);
 static int32 get_rel_data_width(Relation rel, int32 *attr_widths);
@@ -408,7 +408,7 @@ get_relation_info(PlannerInfo *root, Oid relationObjectId, bool inhparent,
        }
 
        /* Collect info about relation's foreign keys, if relevant */
-       get_relation_foreign_keys(root, rel, relation);
+       get_relation_foreign_keys(root, rel, relation, inhparent);
 
        heap_close(relation, NoLock);
 
@@ -433,7 +433,7 @@ get_relation_info(PlannerInfo *root, Oid relationObjectId, bool inhparent,
  */
 static void
 get_relation_foreign_keys(PlannerInfo *root, RelOptInfo *rel,
-                                                 Relation relation)
+                                                 Relation relation, bool inhparent)
 {
        List       *rtable = root->parse->rtable;
        List       *cachedfkeys;
@@ -448,6 +448,15 @@ get_relation_foreign_keys(PlannerInfo *root, RelOptInfo *rel,
                list_length(rtable) < 2)
                return;
 
+       /*
+        * If it's the parent of an inheritance tree, ignore its FKs.  We could
+        * make useful FK-based deductions if we found that all members of the
+        * inheritance tree have equivalent FK constraints, but detecting that
+        * would require code that hasn't been written.
+        */
+       if (inhparent)
+               return;
+
        /*
         * Extract data about relation's FKs from the relcache.  Note that this
         * list belongs to the relcache and might disappear in a cache flush, so
@@ -488,6 +497,9 @@ get_relation_foreign_keys(PlannerInfo *root, RelOptInfo *rel,
                        if (rte->rtekind != RTE_RELATION ||
                                rte->relid != cachedfk->confrelid)
                                continue;
+                       /* Ignore if it's an inheritance parent; doesn't really match */
+                       if (rte->inh)
+                               continue;
                        /* Ignore self-referential FKs; we only care about joins */
                        if (rti == rel->relid)
                                continue;