Location via proxy:   [ UP ]  
[Report a bug]   [Manage cookies]                
Don't build extended statistics on inheritance trees
authorTomas Vondra <tomas.vondra@postgresql.org>
Tue, 30 Jul 2019 17:17:12 +0000 (19:17 +0200)
committerTomas Vondra <tomas.vondra@postgresql.org>
Tue, 30 Jul 2019 17:48:13 +0000 (19:48 +0200)
When performing ANALYZE on inheritance trees, we collect two samples for
each relation - one for the relation alone, and one for the inheritance
subtree (relation and its child relations). And then we build statistics
on each sample, so for each relation we get two sets of statistics.

For regular (per-column) statistics this works fine, because the catalog
includes a flag differentiating statistics built from those two samples.
But we don't have such flag in the extended statistics catalogs, and we
ended up updating the same row twice, triggering this error:

  ERROR:  tuple already updated by self

The simplest solution is to disable extended statistics on inheritance
trees, which is what this commit is doing. In the future we may need to
do something similar to per-column statistics, but that requires adding a
flag to the catalog - and that's not backpatchable. Moreover, the current
selectivity estimation code only works with individual relations, so
building statistics on inheritance trees would be pointless anyway.

Author: Tomas Vondra
Backpatch-to: 10-
Discussion: https://postgr.es/m/20190618231233.GA27470@telsasoft.com
Reported-by: Justin Pryzby
src/backend/commands/analyze.c
src/test/regress/expected/stats_ext.out
src/test/regress/sql/stats_ext.sql

index d7004e53138d80605814f93486a65cf55c28ce21..faabbeb002d75f10b97367c5eef1001beb5fa0c5 100644 (file)
@@ -573,9 +573,15 @@ do_analyze_rel(Relation onerel, VacuumParams *params,
                            thisdata->attr_cnt, thisdata->vacattrstats);
        }
 
-       /* Build extended statistics (if there are any). */
-       BuildRelationExtStatistics(onerel, totalrows, numrows, rows, attr_cnt,
-                                  vacattrstats);
+       /*
+        * Build extended statistics (if there are any).
+        *
+        * For now we only build extended statistics on individual relations,
+        * not for relations representing inheritance trees.
+        */
+       if (!inh)
+           BuildRelationExtStatistics(onerel, totalrows, numrows, rows,
+                                      attr_cnt, vacattrstats);
    }
 
    /*
index 94b8a8f8b85d55fb4c3d49ae188c461d5343202a..d782ebcfeaa8629eba90dbfaf4867a5552cc3e0b 100644 (file)
@@ -103,6 +103,14 @@ ANALYZE ab1 (a);
 WARNING:  statistics object "public.ab1_a_b_stats" could not be computed for relation "public.ab1"
 ANALYZE ab1;
 DROP TABLE ab1;
+-- Ensure we can build statistics for tables with inheritance.
+CREATE TABLE ab1 (a INTEGER, b INTEGER);
+CREATE TABLE ab1c () INHERITS (ab1);
+INSERT INTO ab1 VALUES (1,1);
+CREATE STATISTICS ab1_a_b_stats ON a, b FROM ab1;
+ANALYZE ab1;
+DROP TABLE ab1 CASCADE;
+NOTICE:  drop cascades to table ab1c
 -- Verify supported object types for extended statistics
 CREATE schema tststats;
 CREATE TABLE tststats.t (a int, b int, c text);
index 4bc1536727b46602a9d231248d19b24f54814464..d506e8238c32efd86b8d5053bd494132ef956c28 100644 (file)
@@ -73,6 +73,14 @@ ANALYZE ab1 (a);
 ANALYZE ab1;
 DROP TABLE ab1;
 
+-- Ensure we can build statistics for tables with inheritance.
+CREATE TABLE ab1 (a INTEGER, b INTEGER);
+CREATE TABLE ab1c () INHERITS (ab1);
+INSERT INTO ab1 VALUES (1,1);
+CREATE STATISTICS ab1_a_b_stats ON a, b FROM ab1;
+ANALYZE ab1;
+DROP TABLE ab1 CASCADE;
+
 -- Verify supported object types for extended statistics
 CREATE schema tststats;