Commit ae1b7e29 authored by Tom Lane's avatar Tom Lane

Allow plpgsql function parameter names to be qualified with the function's

name.  With this patch, it is always possible for the user to qualify a
plpgsql variable name if needed to avoid ambiguity.  While there is much more
work to be done in this area, this simple change removes one unnecessary
incompatibility with Oracle.  Per discussion.
parent 9f6f51d5
<!-- $PostgreSQL: pgsql/doc/src/sgml/plpgsql.sgml,v 1.114 2007/07/15 00:45:16 tgl Exp $ --> <!-- $PostgreSQL: pgsql/doc/src/sgml/plpgsql.sgml,v 1.115 2007/07/16 17:01:10 tgl Exp $ -->
<chapter id="plpgsql"> <chapter id="plpgsql">
<title><application>PL/pgSQL</application> - <acronym>SQL</acronym> Procedural Language</title> <title><application>PL/pgSQL</application> - <acronym>SQL</acronym> Procedural Language</title>
...@@ -249,10 +249,23 @@ $$ LANGUAGE plpgsql; ...@@ -249,10 +249,23 @@ $$ LANGUAGE plpgsql;
</programlisting> </programlisting>
</para> </para>
<note>
<para>
There is actually a hidden <quote>outer block</> surrounding the body
of any <application>PL/pgSQL</> function. This block provides the
declarations of the function's parameters (if any), as well as some
special variables such as <literal>FOUND</literal> (see
<xref linkend="plpgsql-statements-diagnostics">). The outer block is
labeled with the function's name, meaning that parameters and special
variables can be qualified with the function's name.
</para>
</note>
<para> <para>
It is important not to confuse the use of It is important not to confuse the use of
<command>BEGIN</>/<command>END</> for grouping statements in <command>BEGIN</>/<command>END</> for grouping statements in
<application>PL/pgSQL</> with the database commands for transaction <application>PL/pgSQL</> with the similarly-named SQL commands
for transaction
control. <application>PL/pgSQL</>'s <command>BEGIN</>/<command>END</> control. <application>PL/pgSQL</>'s <command>BEGIN</>/<command>END</>
are only for grouping; they do not start or end a transaction. are only for grouping; they do not start or end a transaction.
Functions and trigger procedures are always executed within a transaction Functions and trigger procedures are always executed within a transaction
...@@ -370,6 +383,19 @@ BEGIN ...@@ -370,6 +383,19 @@ BEGIN
END; END;
$$ LANGUAGE plpgsql; $$ LANGUAGE plpgsql;
</programlisting> </programlisting>
</para>
<note>
<para>
These two examples are not perfectly equivalent. In the first case,
<literal>subtotal</> could be referenced as
<literal>sales_tax.subtotal</>, but in the second case it could not.
(Had we attached a label to the block, <literal>subtotal</> could
be qualified with that label, instead.)
</para>
</note>
<para>
Some more examples: Some more examples:
<programlisting> <programlisting>
CREATE FUNCTION instr(varchar, integer) RETURNS integer AS $$ CREATE FUNCTION instr(varchar, integer) RETURNS integer AS $$
...@@ -3618,12 +3644,12 @@ a_output := a_output || $$ if v_$$ || referrer_keys.kind || $$ like '$$ ...@@ -3618,12 +3644,12 @@ a_output := a_output || $$ if v_$$ || referrer_keys.kind || $$ like '$$
<listitem> <listitem>
<para> <para>
You cannot use parameter names that are the same as columns If a name used in a SQL command could be either a column name of a
that are referenced in the function. Oracle allows you to do this table or a reference to a variable of the function,
if you qualify the parameter name using <application>PL/SQL</> treats it as a column name, while
<literal>function_name.parameter_name</>. <application>PL/pgSQL</> treats it as a variable name. It's best
In <application>PL/pgSQL</>, you can instead avoid a conflict by to avoid such ambiguities in the first place, but if necessary you
qualifying the column or table name. can fix them by properly qualifying the ambiguous name.
(See <xref linkend="plpgsql-var-subst">.) (See <xref linkend="plpgsql-var-subst">.)
</para> </para>
</listitem> </listitem>
......
...@@ -9,7 +9,7 @@ ...@@ -9,7 +9,7 @@
* *
* *
* IDENTIFICATION * IDENTIFICATION
* $PostgreSQL: pgsql/src/pl/plpgsql/src/gram.y,v 1.103 2007/07/15 02:15:04 tgl Exp $ * $PostgreSQL: pgsql/src/pl/plpgsql/src/gram.y,v 1.104 2007/07/16 17:01:10 tgl Exp $
* *
*------------------------------------------------------------------------- *-------------------------------------------------------------------------
*/ */
...@@ -366,7 +366,7 @@ decl_statement : decl_varname decl_const decl_datatype decl_notnull decl_defval ...@@ -366,7 +366,7 @@ decl_statement : decl_varname decl_const decl_datatype decl_notnull decl_defval
plpgsql_ns_rename($2, $4); plpgsql_ns_rename($2, $4);
} }
| decl_varname opt_scrollable K_CURSOR | decl_varname opt_scrollable K_CURSOR
{ plpgsql_ns_push(NULL); } { plpgsql_ns_push($1.name); }
decl_cursor_args decl_is_for decl_cursor_query decl_cursor_args decl_is_for decl_cursor_query
{ {
PLpgSQL_var *new; PLpgSQL_var *new;
......
...@@ -8,7 +8,7 @@ ...@@ -8,7 +8,7 @@
* *
* *
* IDENTIFICATION * IDENTIFICATION
* $PostgreSQL: pgsql/src/pl/plpgsql/src/pl_comp.c,v 1.116 2007/06/26 16:48:09 alvherre Exp $ * $PostgreSQL: pgsql/src/pl/plpgsql/src/pl_comp.c,v 1.117 2007/07/16 17:01:10 tgl Exp $
* *
*------------------------------------------------------------------------- *-------------------------------------------------------------------------
*/ */
...@@ -306,10 +306,12 @@ do_compile(FunctionCallInfo fcinfo, ...@@ -306,10 +306,12 @@ do_compile(FunctionCallInfo fcinfo,
error_context_stack = &plerrcontext; error_context_stack = &plerrcontext;
/* /*
* Initialize the compiler * Initialize the compiler, particularly the namespace stack. The
* outermost namespace contains function parameters and other special
* variables (such as FOUND), and is named after the function itself.
*/ */
plpgsql_ns_init(); plpgsql_ns_init();
plpgsql_ns_push(NULL); plpgsql_ns_push(NameStr(procStruct->proname));
plpgsql_DumpExecTree = false; plpgsql_DumpExecTree = false;
datums_alloc = 128; datums_alloc = 128;
......
...@@ -8,7 +8,7 @@ ...@@ -8,7 +8,7 @@
* *
* *
* IDENTIFICATION * IDENTIFICATION
* $PostgreSQL: pgsql/src/pl/plpgsql/src/pl_funcs.c,v 1.60 2007/07/15 02:15:04 tgl Exp $ * $PostgreSQL: pgsql/src/pl/plpgsql/src/pl_funcs.c,v 1.61 2007/07/16 17:01:11 tgl Exp $
* *
*------------------------------------------------------------------------- *-------------------------------------------------------------------------
*/ */
...@@ -147,15 +147,14 @@ plpgsql_ns_setlocal(bool flag) ...@@ -147,15 +147,14 @@ plpgsql_ns_setlocal(bool flag)
* ---------- * ----------
*/ */
void void
plpgsql_ns_push(char *label) plpgsql_ns_push(const char *label)
{ {
PLpgSQL_ns *new; PLpgSQL_ns *new;
if (label == NULL) if (label == NULL)
label = ""; label = "";
new = palloc(sizeof(PLpgSQL_ns)); new = palloc0(sizeof(PLpgSQL_ns));
memset(new, 0, sizeof(PLpgSQL_ns));
new->upper = ns_current; new->upper = ns_current;
ns_current = new; ns_current = new;
...@@ -224,7 +223,7 @@ plpgsql_ns_additem(int itemtype, int itemno, const char *name) ...@@ -224,7 +223,7 @@ plpgsql_ns_additem(int itemtype, int itemno, const char *name)
* ---------- * ----------
*/ */
PLpgSQL_nsitem * PLpgSQL_nsitem *
plpgsql_ns_lookup(char *name, char *label) plpgsql_ns_lookup(const char *name, const char *label)
{ {
PLpgSQL_ns *ns; PLpgSQL_ns *ns;
int i; int i;
...@@ -236,11 +235,11 @@ plpgsql_ns_lookup(char *name, char *label) ...@@ -236,11 +235,11 @@ plpgsql_ns_lookup(char *name, char *label)
{ {
for (ns = ns_current; ns != NULL; ns = ns->upper) for (ns = ns_current; ns != NULL; ns = ns->upper)
{ {
if (!strcmp(ns->items[0]->name, label)) if (strcmp(ns->items[0]->name, label) == 0)
{ {
for (i = 1; i < ns->items_used; i++) for (i = 1; i < ns->items_used; i++)
{ {
if (!strcmp(ns->items[i]->name, name)) if (strcmp(ns->items[i]->name, name) == 0)
return ns->items[i]; return ns->items[i];
} }
return NULL; /* name not found in specified label */ return NULL; /* name not found in specified label */
...@@ -254,7 +253,7 @@ plpgsql_ns_lookup(char *name, char *label) ...@@ -254,7 +253,7 @@ plpgsql_ns_lookup(char *name, char *label)
*/ */
for (ns = ns_current; ns != NULL; ns = ns->upper) for (ns = ns_current; ns != NULL; ns = ns->upper)
{ {
if (!strcmp(ns->items[0]->name, name)) if (strcmp(ns->items[0]->name, name) == 0)
return ns->items[0]; return ns->items[0];
} }
...@@ -265,7 +264,7 @@ plpgsql_ns_lookup(char *name, char *label) ...@@ -265,7 +264,7 @@ plpgsql_ns_lookup(char *name, char *label)
{ {
for (i = 1; i < ns->items_used; i++) for (i = 1; i < ns->items_used; i++)
{ {
if (!strcmp(ns->items[i]->name, name)) if (strcmp(ns->items[i]->name, name) == 0)
return ns->items[i]; return ns->items[i];
} }
if (ns_localmode) if (ns_localmode)
...@@ -288,14 +287,13 @@ plpgsql_ns_rename(char *oldname, char *newname) ...@@ -288,14 +287,13 @@ plpgsql_ns_rename(char *oldname, char *newname)
int i; int i;
/* /*
* Lookup name in the namestack; do the lookup in the current namespace * Lookup name in the namestack
* only.
*/ */
for (ns = ns_current; ns != NULL; ns = ns->upper) for (ns = ns_current; ns != NULL; ns = ns->upper)
{ {
for (i = 1; i < ns->items_used; i++) for (i = 1; i < ns->items_used; i++)
{ {
if (!strcmp(ns->items[i]->name, oldname)) if (strcmp(ns->items[i]->name, oldname) == 0)
{ {
newitem = palloc(sizeof(PLpgSQL_nsitem) + strlen(newname)); newitem = palloc(sizeof(PLpgSQL_nsitem) + strlen(newname));
newitem->itemtype = ns->items[i]->itemtype; newitem->itemtype = ns->items[i]->itemtype;
......
...@@ -8,7 +8,7 @@ ...@@ -8,7 +8,7 @@
* *
* *
* IDENTIFICATION * IDENTIFICATION
* $PostgreSQL: pgsql/src/pl/plpgsql/src/plpgsql.h,v 1.89 2007/07/15 02:15:04 tgl Exp $ * $PostgreSQL: pgsql/src/pl/plpgsql/src/plpgsql.h,v 1.90 2007/07/16 17:01:11 tgl Exp $
* *
*------------------------------------------------------------------------- *-------------------------------------------------------------------------
*/ */
...@@ -770,10 +770,10 @@ extern char *plpgsql_dstring_get(PLpgSQL_dstring *ds); ...@@ -770,10 +770,10 @@ extern char *plpgsql_dstring_get(PLpgSQL_dstring *ds);
*/ */
extern void plpgsql_ns_init(void); extern void plpgsql_ns_init(void);
extern bool plpgsql_ns_setlocal(bool flag); extern bool plpgsql_ns_setlocal(bool flag);
extern void plpgsql_ns_push(char *label); extern void plpgsql_ns_push(const char *label);
extern void plpgsql_ns_pop(void); extern void plpgsql_ns_pop(void);
extern void plpgsql_ns_additem(int itemtype, int itemno, const char *name); extern void plpgsql_ns_additem(int itemtype, int itemno, const char *name);
extern PLpgSQL_nsitem *plpgsql_ns_lookup(char *name, char *nsname); extern PLpgSQL_nsitem *plpgsql_ns_lookup(const char *name, const char *nsname);
extern void plpgsql_ns_rename(char *oldname, char *newname); extern void plpgsql_ns_rename(char *oldname, char *newname);
/* ---------- /* ----------
......
...@@ -3051,3 +3051,31 @@ select * from sc_test(); ...@@ -3051,3 +3051,31 @@ select * from sc_test();
(3 rows) (3 rows)
drop function sc_test(); drop function sc_test();
-- test qualified variable names
create function pl_qual_names (param1 int) returns void as $$
<<outerblock>>
declare
param1 int := 1;
begin
<<innerblock>>
declare
param1 int := 2;
begin
raise notice 'param1 = %', param1;
raise notice 'pl_qual_names.param1 = %', pl_qual_names.param1;
raise notice 'outerblock.param1 = %', outerblock.param1;
raise notice 'innerblock.param1 = %', innerblock.param1;
end;
end;
$$ language plpgsql;
select pl_qual_names(42);
NOTICE: param1 = 2
NOTICE: pl_qual_names.param1 = 42
NOTICE: outerblock.param1 = 1
NOTICE: innerblock.param1 = 2
pl_qual_names
---------------
(1 row)
drop function pl_qual_names(int);
...@@ -2535,3 +2535,25 @@ select * from sc_test(); ...@@ -2535,3 +2535,25 @@ select * from sc_test();
drop function sc_test(); drop function sc_test();
-- test qualified variable names
create function pl_qual_names (param1 int) returns void as $$
<<outerblock>>
declare
param1 int := 1;
begin
<<innerblock>>
declare
param1 int := 2;
begin
raise notice 'param1 = %', param1;
raise notice 'pl_qual_names.param1 = %', pl_qual_names.param1;
raise notice 'outerblock.param1 = %', outerblock.param1;
raise notice 'innerblock.param1 = %', innerblock.param1;
end;
end;
$$ language plpgsql;
select pl_qual_names(42);
drop function pl_qual_names(int);
Markdown is supported
0% or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment