Predictable plans with pg_hint_plan full hinting

Franck Pachot - Jun 9 '22 - - Dev Community

Any SQL database provides hints to give more control on the optimizer, either to troubleshoot and understand the query planner decision, to workaround impossible cost estimation, or to fix a plan for stability.

Yes, PostgreSQL has query planner hints

With PostgreSQL, the extension to do it, pg_hint_plan is really good, but not widely used because not included in the core, not even in contrib. The consequence is that people install it only when needing it, without the time to learn how to hint correctly, and may think that "my hint is not used" and give up.

Better learn before you need it

To learn it before having the need for it, you can use this dockerfile to build a PostgreSQL image with pg_hint_plan. Or better, use the YugabyteDB image or free managed with cloud shell. As bad execution plans are worse in high performance distributed SQL databases, YugabyteDB has pg_hint_plan installed and enabled by default.

Example on YugabyteDB

I create 3 simple tables:

create table table_a( id int primary key) ;
create table table_b( id int primary key) ;
create table table_c( id int primary key) ;
Enter fullscreen mode Exit fullscreen mode

I query the execution plan for a join between those two tables:

yugabyte=# explain (costs off, timing off)
           select * from table_a a join table_b b using(id);

                   QUERY PLAN
------------------------------------------------
 Nested Loop
   ->  Seq Scan on table_a
   ->  Index Scan using table_b_pkey on table_b
         Index Cond: (id = table_a.id)
Enter fullscreen mode Exit fullscreen mode

The query planner has chosen a Nested Loop with table_a being the outer table and table_b the inner table. It is executed as an outer loop on table_a rows and, for each one, an inner loop on table_b matching rows. The outer table is also called 'left', and inner 'right', because we are used to read from left to right. Explain displays it from top to bottom with a join operation and two child nodes: first is the outer table, second is the inner table.

Leading the join order

Now, let's say that, for whatever reason (understanding, troubleshooting, fixing), I want the opposite order, starting to scan table_b and from it access to table_a. I'll use the Leading hint to define by which table the planner has to start, and which table to join next. Note that when I say "table" it is actually the alias because one table may have different roles in a query block.

A common error is using the Leading( b a ) hint like this:

yugabyte=# explain (costs off, timing off)
          /*+ Leading( b a ) */ 
          select * from table_a a join table_b b using(id);

                    QUERY PLAN
--------------------------------------------------
 Nested Loop
   ->  Seq Scan on table_a a
   ->  Index Scan using table_b_pkey on table_b b
         Index Cond: (id = a.id)
Enter fullscreen mode Exit fullscreen mode

As you can see, nothing has changed in my plan. The reason is that the Leading hint with a simple list of tables defines in which order the query planner will consider each table, but doesn't define the join direction, which is still left to the query planner estimations. This Leading syntax just says "start by joining b and a", which is useless here because there are no other solution with two tables only. There's no error reported in the verbose log (see below how to enable it) because the hint was used, even if the intention of the user was different.

Outer to Inner join direction

From a relational point of view, the join operation is commutative. But the implementation of the join method (Nested Loop, Merge Join, Hash Join) is not.

In any query, the tables will be joined two by two, visible as two child operations on the join node displayed by the explain tree. Each one is actually a table or index scan, or the result from a previous join. For each join, there are two possible directions, defining which one is the outer (or left, on top) or the inner (or right, on second, and last, position). This choice is important because the efficiency of a join method, like Nested Loop, depends on the number of rows of the outer table, and the access path to the inner one.

Different syntax for join direction

In order to define the join direction, you need more parentheses to order each join pairs, where the left one is the outer and the right one the inner.

With those two tables, this is as simple as defining (b a) instead of b a and this is put within the Leading() arguments:

yugabyte=# explain (costs off, timing off)
          /*+ Leading( ( b a ) ) */ 
          select * from table_a a join table_b b using(id);

                    QUERY PLAN
--------------------------------------------------
 Nested Loop
   ->  Seq Scan on table_b b
   ->  Index Scan using table_a_pkey on table_a a
         Index Cond: (id = b.id)
Enter fullscreen mode Exit fullscreen mode

This is the expected plan, joining table_a and table_b but with the table_b being the outer (left, first child) and table_a the inner one.

Leading hint for 3 tables

This is more visible with three tables where this (a b) pair is the outer table within the ( ( b a ) c ) pair:

yugabyte=# explain (costs off, timing off)
          /*+ Leading( ( ( b a ) c ) ) */
          select * from table_a a join table_b b using(id)
                                  join table_c c using(id);

                       QUERY PLAN
--------------------------------------------------------
 Nested Loop
   ->  Nested Loop
         ->  Seq Scan on table_b b
         ->  Index Scan using table_a_pkey on table_a a
               Index Cond: (id = b.id)
   ->  Index Scan using table_c_pkey on table_c c
         Index Cond: (id = a.id)
Enter fullscreen mode Exit fullscreen mode

Here I introduced a new pair of join, with the left one being the result of the join between table_a and table_b, hinted with the same direction as above, with (a b), and the right-side table is table_c with ( (a b) c). This adds table_c as the inner table, accessed for each row from the outer table, which is the result of the previous join between table_a and table_b.

Swapping join inputs

Having table_c as the inner table is one possibility, but we may want it to be the outer table by putting it on the left side with ( c (b a) ) instead of ( ( b a ) c ):

yugabyte=# explain (costs off, timing off)
          /*+ Leading( ( c (b a) ) ) */ 
          select * from table_a a join table_b b using(id)
                                  join table_c c using(id);

                       QUERY PLAN
--------------------------------------------------------
 Hash Join
   Hash Cond: (c.id = a.id)
   ->  Seq Scan on table_c c
   ->  Hash
         ->  Nested Loop
               ->  Seq Scan on table_b b
               ->  Index Scan using table_a_pkey on table_a a
                     Index Cond: (id = b.id)
Enter fullscreen mode Exit fullscreen mode

When table_c was the inner table, the query planner had chosen a Nested Loop because there's a fast access in the inner loop, with the primary key index on table_c. As the inner loop of a Nested loop is executed many times, it is efficient only with a permanent range or point access structure, sorted or hashed, like an index. However, when the result of another join is the inner table, there's no permanent structure to access to specific rows in it. But the SQL executor can create a temporary one for the duration of the query. This is why the query planner decides to do a Hash Join, creating the Hash structure on the result of the previous join, to get fast access to this inner row source.

Join method: Nested Loop, Hash or Merge join

When you force a join order you probably also want to force the join method. Remember that, when you need to hint a query, the reason is probably that you don't trust the query planner estimations. If they were bad to get the right join order, then they are probably bad for the join method as well. Even if you see the expected join method above, nothing guarantees that it will always be this one on new data. But there are hints for the join method as well.

The plan above could be fixed by adding a HashJoin() hint. The following, with NestLoop() will force a Nested Loop from table_c to the result of the join between table_a and table_b:

yugabyte=# explain (costs off, timing off)
          /*+ Leading( ( c (b a) ) ) NestLoop(a b c) */ 
          select * from table_a a join table_b b using(id)
                                  join table_c c using(id);

                          QUERY PLAN
--------------------------------------------------------------
 Nested Loop
   Join Filter: (a.id = c.id)
   ->  Seq Scan on table_c c
   ->  Materialize
         ->  Nested Loop
               ->  Seq Scan on table_b b
               ->  Index Scan using table_a_pkey on table_a a
                     Index Cond: (id = b.id)
Enter fullscreen mode Exit fullscreen mode

I have written NestLoop(a b c) in alphabetical order to show that it doesn't matter. This just says that the last join on those three tables must be a Nested Loop. By convention, I prefer to write it as NestLoop(b a c) but it doesn't matter. Note that here, nothing guarantees the Nested Loop for the first join beween table_a and table_b. I should add NestLoop(b a) for this.

An Hint for Hinting: count the parentheses

Note that if I had defined Leading(b a c) as I did at the begining of this post, it would have left many possibilities for the query planner like ((b a)c), ((a b) c), (c(a b)) or ((a b)c). And, in addition to that, two join methods for each one.

For full hinting between n tables, you need to have n-1 nested pairs in Leading(). And each pair is a (). That's a lot of parentheses. By the way, do you know that the first implementation of POSTGRES had some code in LISP?

Back to our hints, once the Leading() is defined with all those n-1 nested pairs, there is one join method to define for each pair (NestLoop(), HashJoin() or MergeJoin). The one at position i (from 1 to n-1) will have i+1 arguments.

In addition to that, to fully define the execution plan, the access for each of n tables must be one of the scan method, like SeqScan(), IndexScan(), IndexOnlyScan() or BitmapScan(). In YugabyteDB, which uses a storage based on PostgreSQL tuples rather than pages, you don't see BitmapScan.

For n tables, you will have in total 2*n hints. And you can also count the opening parentheses, which equals the closing ones, and should find 6*n-2 of them for full hinting.

The last plan above is fully hinted, which means not leaving any other join and access decision to the query planner, by:

/*+ 
    Leading( ( c (b a) ) )
    NestLoop(b a) NestLoop(b a c) 
    SeqScan(b) IndexScan(a table_a_pkey) SeqScan(c)
*/ 
Enter fullscreen mode Exit fullscreen mode

Verification and Troubleshooting

Some hint error syntax will raise a warning. To go further, as mentioned in the previous post, you can get more info with:

set pg_hint_plan.debug_print=verbose;
set client_min_messages = log;
Enter fullscreen mode Exit fullscreen mode

Here is my fully hinted query:

yugabyte=# explain (costs off, timing off)
/*+ 
    Leading( ( c (b a) ) )
    NestLoop(b a) NestLoop(b a c) 
    SeqScan(b) IndexScan(a table_a_pkey) SeqScan(c)
*/ 
select * from table_a a join table_b b using(id)
                        join table_c c using(id);

LOG:  pg_hint_plan[qno=0xf]: planner
LOG:  pg_hint_plan[qno=0xf]: setup_hint_enforcement index deletion: relation=16542(table_a), inhparent=0, current_hint_state=0x3a1c20c8, hint_inhibit_level=0, scanmask=0x2
LOG:  pg_hint_plan[qno=0xf]: setup_hint_enforcement index deletion: relation=16547(table_b), inhparent=0, current_hint_state=0x3a1c20c8, hint_inhibit_level=0, scanmask=0x1
LOG:  pg_hint_plan[qno=0xf]: setup_hint_enforcement index deletion: relation=16552(table_c), inhparent=0, current_hint_state=0x3a1c20c8, hint_inhibit_level=0, scanmask=0x1
LOG:  pg_hint_plan[qno=0xf]: HintStateDump: {used hints:IndexScan(a table_a_pkey)SeqScan(b)SeqScan(c)NestLoop(a b)NestLoop(a b c)Leading((c (b a)))}, {not used hints:(none)}, {duplicate hints:(none)}, {error hints:(none)}

                          QUERY PLAN
--------------------------------------------------------------
 Nested Loop
   Join Filter: (a.id = c.id)
   ->  Seq Scan on table_c c
   ->  Materialize
         ->  Nested Loop
               ->  Seq Scan on table_b b
               ->  Index Scan using table_a_pkey on table_a a
                     Index Cond: (id = b.id)
(8 rows)
Enter fullscreen mode Exit fullscreen mode

The important piece is the HintStateDump which will warn you if some hints were not used, are duplicated, or with error. The used hints is a rewrite of your hints that were used during query planning. You can see here that my NestLoop(b a) and NestLoop(b a c) were rewritten to NestLoop(a b) and NestLoop(a b c) which proves that the order doesn't matter here. There are stored internally in alphabetical order.

The setup_hint_enforcement index deletion is just a trace that indexes are removed from the list of possible ones: totally for SeqScan() (scanmask=0x1) or partially for IndexScan() (scanmask=0x2). When partially, only the one named (like table_a_pkey here with IndexScan()) is kept, the ones matching a pattern with IndexScanRegexp().

Note that if you are only interested by the HintStateDump the debug level on is sufficient. Here is an exhibit for different reporting:

yugabyte=# set pg_hint_plan.debug_print='on';
SET

yugabyte=# select /*+ 
   Leading(x) IndexScan(a) IndexScan(b) SeqScan(b)
   Set(random_page_cost 1e42) xxx() Set(seq_page_cost 0.42) 
*/ * from table_a a;

INFO:  pg_hint_plan: hint syntax error at or near "Leading(x) IndexScan(a) IndexScan(b) SeqScan(b)
   Set(random_page_cost 1e42) xxx() Set(seq_page_cost 0.42)
"
DETAIL:  Leading hint requires at least two relations.
INFO:  pg_hint_plan: hint syntax error at or near "xxx() Set(seq_page_cost 0.42)
"
DETAIL:  Unrecognized hint keyword "xxx".
INFO:  pg_hint_plan: hint syntax error at or near "IndexScan(b) SeqScan(b)
   Set(random_page_cost 1e42) xxx() Set(seq_page_cost 0.42)
"
DETAIL:  Conflict scan method hint.
LOG:  pg_hint_plan:
used hint:
IndexScan(a)
Set(random_page_cost 1e42)
not used hint:
SeqScan(b)
duplication hint:
IndexScan(b)
error hint:
Leading(x)

 id
----
(0 rows)
Enter fullscreen mode Exit fullscreen mode

Mentioning two conflicting scan methods for the same alias has used the first one IndexScan(a) and discarded the other SeqScan(a) as duplication hint. A syntax error in a valid hint, like Leading with only one alias, is detected as error hint. An invalid hint, like xxx here has stopped the parsing and the valid Set(seq_page_cost 0.42) after it is ignored.

Other hints

There are other hints to choose the index with a smarter way. Don't forget that index names can change, which is transparent for SQL, but using hints break the logical-physical independence as you have to name the indexes. There is also the Rows hint that is smarter in some cases, fixing the estimations rather than forcing anything, but it doesn't fix a plan. This is used to add knowledge about data correlation between tables, which is a common source of cost misestimate.

There are also the opposite hints like NoSeqScan to allow any scan except SeqScan but remember that hints are already a negation. They are actually implemented by raising the cost of the operation to its maximum value.

You can also set parameters at statement level rather than session level with the Set hint. Useful to restrict the scope of a workaround to the scope of the problem, but I don't think it is needed to fix a plan. For example, my fully hinted query above will still show the same plan in a session where I set enable_indexscan=off; because the IndexScan hint internally sets it to on when specified for a table.

Core message

The most important decision is the join order, in the Leading() hint and to define the direction its arguments should be a list of nested ordered pairs rather than an ordered list. Do not think that, when you see the right execution plan after adding some hints, it means that you have fixed a plan. Without full hinting (the 2*n hints with 6*n-2 parenthesis pairs in total for n tables involved), the query planner may still come with another idea, and that may happen next Sunday at 2 a.m.

Here is a PostgresWorld Webinar on this:

. . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . .