]> granicus.if.org Git - postgresql/commitdiff
Fix plpgsql's reporting of plan-time errors in possibly-simple expressions.
authorTom Lane <tgl@sss.pgh.pa.us>
Thu, 31 Jan 2013 01:02:33 +0000 (20:02 -0500)
committerTom Lane <tgl@sss.pgh.pa.us>
Thu, 31 Jan 2013 01:02:33 +0000 (20:02 -0500)
exec_simple_check_plan and exec_eval_simple_expr attempted to call
GetCachedPlan directly.  This meant that if an error was thrown during
planning, the resulting context traceback would not include the line
normally contributed by _SPI_error_callback.  This is already inconsistent,
but just to be really odd, a re-execution of the very same expression
*would* show the additional context line, because we'd already have cached
the plan and marked the expression as non-simple.

The problem is easy to demonstrate in 9.2 and HEAD because planning of a
cached plan doesn't occur at all until GetCachedPlan is done.  In earlier
versions, it could only be an issue if initial planning had succeeded, then
a replan was forced (already somewhat improbable for a simple expression),
and the replan attempt failed.  Since the issue is mainly cosmetic in older
branches anyway, it doesn't seem worth the risk of trying to fix it there.
It is worth fixing in 9.2 since the instability of the context printout can
affect the results of GET STACKED DIAGNOSTICS, as per a recent discussion
on pgsql-novice.

To fix, introduce a SPI function that wraps GetCachedPlan while installing
the correct callback function.  Use this instead of calling GetCachedPlan
directly from plpgsql.

Also introduce a wrapper function for extracting a SPI plan's
CachedPlanSource list.  This lets us stop including spi_priv.h in
pl_exec.c, which was never a very good idea from a modularity standpoint.

In passing, fix a similar inconsistency that could occur in SPI_cursor_open,
which was also calling GetCachedPlan without setting up a context callback.

src/backend/executor/spi.c
src/include/executor/spi.h
src/pl/plpgsql/src/pl_exec.c
src/test/regress/expected/plpgsql.out
src/test/regress/sql/plpgsql.sql

index da40f8207c19f98c80730c7d48b1c4daf6a164da..96465e161e1bf24618438eea35998410be66517d 100644 (file)
@@ -1131,6 +1131,7 @@ SPI_cursor_open_internal(const char *name, SPIPlanPtr plan,
        Snapshot        snapshot;
        MemoryContext oldcontext;
        Portal          portal;
+       ErrorContextCallback spierrcontext;
 
        /*
         * Check that the plan is something the Portal code will special-case as
@@ -1180,6 +1181,15 @@ SPI_cursor_open_internal(const char *name, SPIPlanPtr plan,
        query_string = MemoryContextStrdup(PortalGetHeapMemory(portal),
                                                                           plansource->query_string);
 
+       /*
+        * Setup error traceback support for ereport(), in case GetCachedPlan
+        * throws an error.
+        */
+       spierrcontext.callback = _SPI_error_callback;
+       spierrcontext.arg = (void *) plansource->query_string;
+       spierrcontext.previous = error_context_stack;
+       error_context_stack = &spierrcontext;
+
        /*
         * Note: for a saved plan, we mustn't have any failure occur between
         * GetCachedPlan and PortalDefineQuery; that would result in leaking our
@@ -1190,6 +1200,9 @@ SPI_cursor_open_internal(const char *name, SPIPlanPtr plan,
        cplan = GetCachedPlan(plansource, paramLI, false);
        stmt_list = cplan->stmt_list;
 
+       /* Pop the error context stack */
+       error_context_stack = spierrcontext.previous;
+
        if (!plan->saved)
        {
                /*
@@ -1551,6 +1564,65 @@ SPI_result_code_string(int code)
        return buf;
 }
 
+/*
+ * SPI_plan_get_plan_sources --- get a SPI plan's underlying list of
+ * CachedPlanSources.
+ *
+ * This is exported so that pl/pgsql can use it (this beats letting pl/pgsql
+ * look directly into the SPIPlan for itself).  It's not documented in
+ * spi.sgml because we'd just as soon not have too many places using this.
+ */
+List *
+SPI_plan_get_plan_sources(SPIPlanPtr plan)
+{
+       Assert(plan->magic == _SPI_PLAN_MAGIC);
+       return plan->plancache_list;
+}
+
+/*
+ * SPI_plan_get_cached_plan --- get a SPI plan's generic CachedPlan,
+ * if the SPI plan contains exactly one CachedPlanSource.  If not,
+ * return NULL.  Caller is responsible for doing ReleaseCachedPlan().
+ *
+ * This is exported so that pl/pgsql can use it (this beats letting pl/pgsql
+ * look directly into the SPIPlan for itself).  It's not documented in
+ * spi.sgml because we'd just as soon not have too many places using this.
+ */
+CachedPlan *
+SPI_plan_get_cached_plan(SPIPlanPtr plan)
+{
+       CachedPlanSource *plansource;
+       CachedPlan *cplan;
+       ErrorContextCallback spierrcontext;
+
+       Assert(plan->magic == _SPI_PLAN_MAGIC);
+
+       /* Can't support one-shot plans here */
+       if (plan->oneshot)
+               return NULL;
+
+       /* Must have exactly one CachedPlanSource */
+       if (list_length(plan->plancache_list) != 1)
+               return NULL;
+       plansource = (CachedPlanSource *) linitial(plan->plancache_list);
+
+       /* Setup error traceback support for ereport() */
+       spierrcontext.callback = _SPI_error_callback;
+       spierrcontext.arg = (void *) plansource->query_string;
+       spierrcontext.previous = error_context_stack;
+       error_context_stack = &spierrcontext;
+
+       /* Get the generic plan for the query */
+       cplan = GetCachedPlan(plansource, NULL, plan->saved);
+       Assert(cplan == plansource->gplan);
+
+       /* Pop the error context stack */
+       error_context_stack = spierrcontext.previous;
+
+       return cplan;
+}
+
+
 /* =================== private functions =================== */
 
 /*
index cfbaa14213528e952b2bb0bbc8c8931fa0ba9f8f..4e01c9c0f90293510e073e77a004a91b16452dcc 100644 (file)
@@ -103,6 +103,9 @@ extern bool SPI_is_cursor_plan(SPIPlanPtr plan);
 extern bool SPI_plan_is_valid(SPIPlanPtr plan);
 extern const char *SPI_result_code_string(int code);
 
+extern List *SPI_plan_get_plan_sources(SPIPlanPtr plan);
+extern CachedPlan *SPI_plan_get_cached_plan(SPIPlanPtr plan);
+
 extern HeapTuple SPI_copytuple(HeapTuple tuple);
 extern HeapTupleHeader SPI_returntuple(HeapTuple tuple, TupleDesc tupdesc);
 extern HeapTuple SPI_modifytuple(Relation rel, HeapTuple tuple, int natts,
index 8ca791ce3f42486a8dc47634397600d4b2b7c915..fbc8e78ec0970a9a7050607cc9bf500900033d13 100644 (file)
@@ -21,7 +21,7 @@
 #include "access/tupconvert.h"
 #include "catalog/pg_proc.h"
 #include "catalog/pg_type.h"
-#include "executor/spi_priv.h"
+#include "executor/spi.h"
 #include "funcapi.h"
 #include "miscadmin.h"
 #include "nodes/nodeFuncs.h"
@@ -3029,7 +3029,7 @@ exec_stmt_execsql(PLpgSQL_execstate *estate,
 
                exec_prepare_plan(estate, expr, 0);
                stmt->mod_stmt = false;
-               foreach(l, expr->plan->plancache_list)
+               foreach(l, SPI_plan_get_plan_sources(expr->plan))
                {
                        CachedPlanSource *plansource = (CachedPlanSource *) lfirst(l);
                        ListCell   *l2;
@@ -4846,10 +4846,11 @@ loop_exit:
  *
  * It is possible though unlikely for a simple expression to become non-simple
  * (consider for example redefining a trivial view).  We must handle that for
- * correctness; fortunately it's normally inexpensive to do GetCachedPlan on a
- * simple expression.  We do not consider the other direction (non-simple
- * expression becoming simple) because we'll still give correct results if
- * that happens, and it's unlikely to be worth the cycles to check.
+ * correctness; fortunately it's normally inexpensive to call
+ * SPI_plan_get_cached_plan for a simple expression.  We do not consider the
+ * other direction (non-simple expression becoming simple) because we'll still
+ * give correct results if that happens, and it's unlikely to be worth the
+ * cycles to check.
  *
  * Note: if pass-by-reference, the result is in the eval_econtext's
  * temporary memory context.  It will be freed when exec_eval_cleanup
@@ -4865,7 +4866,6 @@ exec_eval_simple_expr(PLpgSQL_execstate *estate,
 {
        ExprContext *econtext = estate->eval_econtext;
        LocalTransactionId curlxid = MyProc->lxid;
-       CachedPlanSource *plansource;
        CachedPlan *cplan;
        ParamListInfo paramLI;
        PLpgSQL_expr *save_cur_expr;
@@ -4885,16 +4885,16 @@ exec_eval_simple_expr(PLpgSQL_execstate *estate,
 
        /*
         * Revalidate cached plan, so that we will notice if it became stale. (We
-        * need to hold a refcount while using the plan, anyway.)  Note that even
-        * if replanning occurs, the length of plancache_list can't change, since
-        * it is a property of the raw parsetree generated from the query text.
+        * need to hold a refcount while using the plan, anyway.)
         */
-       Assert(list_length(expr->plan->plancache_list) == 1);
-       plansource = (CachedPlanSource *) linitial(expr->plan->plancache_list);
+       cplan = SPI_plan_get_cached_plan(expr->plan);
 
-       /* Get the generic plan for the query */
-       cplan = GetCachedPlan(plansource, NULL, true);
-       Assert(cplan == plansource->gplan);
+       /*
+        * We can't get a failure here, because the number of CachedPlanSources in
+        * the SPI plan can't change from what exec_simple_check_plan saw; it's a
+        * property of the raw parsetree generated from the query text.
+        */
+       Assert(cplan != NULL);
 
        if (cplan->generation != expr->expr_simple_generation)
        {
@@ -5702,6 +5702,7 @@ exec_simple_check_node(Node *node)
 static void
 exec_simple_check_plan(PLpgSQL_expr *expr)
 {
+       List       *plansources;
        CachedPlanSource *plansource;
        Query      *query;
        CachedPlan *cplan;
@@ -5717,9 +5718,10 @@ exec_simple_check_plan(PLpgSQL_expr *expr)
        /*
         * We can only test queries that resulted in exactly one CachedPlanSource
         */
-       if (list_length(expr->plan->plancache_list) != 1)
+       plansources = SPI_plan_get_plan_sources(expr->plan);
+       if (list_length(plansources) != 1)
                return;
-       plansource = (CachedPlanSource *) linitial(expr->plan->plancache_list);
+       plansource = (CachedPlanSource *) linitial(plansources);
 
        /*
         * Do some checking on the analyzed-and-rewritten form of the query. These
@@ -5777,8 +5779,10 @@ exec_simple_check_plan(PLpgSQL_expr *expr)
         */
 
        /* Get the generic plan for the query */
-       cplan = GetCachedPlan(plansource, NULL, true);
-       Assert(cplan == plansource->gplan);
+       cplan = SPI_plan_get_cached_plan(expr->plan);
+
+       /* Can't fail, because we checked for a single CachedPlanSource above */
+       Assert(cplan != NULL);
 
        /* Share the remaining work with recheck code path */
        exec_simple_recheck_plan(expr, cplan);
index fb47fa7d45cff0ae7f5f4b3bc49d7919d8e23184..0582e6275ed7cc0aeb28bf7e75e7b8fe3028375a 100644 (file)
@@ -4258,6 +4258,21 @@ select error2('public.stuffs');
 rollback;
 drop function error2(p_name_table text);
 drop function error1(text);
+-- Test for consistent reporting of error context
+create function fail() returns int language plpgsql as $$
+begin
+  return 1/0;
+end
+$$;
+select fail();
+ERROR:  division by zero
+CONTEXT:  SQL statement "SELECT 1/0"
+PL/pgSQL function fail() line 3 at RETURN
+select fail();
+ERROR:  division by zero
+CONTEXT:  SQL statement "SELECT 1/0"
+PL/pgSQL function fail() line 3 at RETURN
+drop function fail();
 -- Test handling of string literals.
 set standard_conforming_strings = off;
 create or replace function strtest() returns text as $$
index 2b60b678af33717531afbaa40c8af29fafaeaec8..83385090f444f50be8067dd8f2128e191990b124 100644 (file)
@@ -3425,6 +3425,19 @@ rollback;
 drop function error2(p_name_table text);
 drop function error1(text);
 
+-- Test for consistent reporting of error context
+
+create function fail() returns int language plpgsql as $$
+begin
+  return 1/0;
+end
+$$;
+
+select fail();
+select fail();
+
+drop function fail();
+
 -- Test handling of string literals.
 
 set standard_conforming_strings = off;