Skip to content
Projects
Groups
Snippets
Help
Loading...
Help
Support
Keyboard shortcuts
?
Submit feedback
Contribute to GitLab
Sign in
Toggle navigation
P
Postgres FD Implementation
Project overview
Project overview
Details
Activity
Releases
Repository
Repository
Files
Commits
Branches
Tags
Contributors
Graph
Compare
Issues
0
Issues
0
List
Boards
Labels
Milestones
Merge Requests
0
Merge Requests
0
CI / CD
CI / CD
Pipelines
Jobs
Schedules
Analytics
Analytics
CI / CD
Repository
Value Stream
Wiki
Wiki
Snippets
Snippets
Members
Members
Collapse sidebar
Close sidebar
Activity
Graph
Create a new issue
Jobs
Commits
Issue Boards
Open sidebar
Abuhujair Javed
Postgres FD Implementation
Commits
470a1048
Commit
470a1048
authored
Sep 01, 2002
by
Tom Lane
Browse files
Options
Browse Files
Download
Email Patches
Plain Diff
plpgsql functions can return RECORD, per Neil Conway.
parent
19032215
Changes
7
Hide whitespace changes
Inline
Side-by-side
Showing
7 changed files
with
123 additions
and
17 deletions
+123
-17
doc/src/sgml/plpgsql.sgml
doc/src/sgml/plpgsql.sgml
+1
-3
doc/src/sgml/xfunc.sgml
doc/src/sgml/xfunc.sgml
+22
-1
src/pl/plpgsql/src/gram.y
src/pl/plpgsql/src/gram.y
+1
-2
src/pl/plpgsql/src/pl_comp.c
src/pl/plpgsql/src/pl_comp.c
+7
-7
src/pl/plpgsql/src/pl_exec.c
src/pl/plpgsql/src/pl_exec.c
+4
-4
src/test/regress/expected/plpgsql.out
src/test/regress/expected/plpgsql.out
+53
-0
src/test/regress/sql/plpgsql.sql
src/test/regress/sql/plpgsql.sql
+35
-0
No files found.
doc/src/sgml/plpgsql.sgml
View file @
470a1048
<
!--
<
!--
$
Header
:
/
cvsroot
/
pgsql
/
doc
/
src
/
sgml
/
plpgsql
.
sgml
,
v
1.
5
2002
/
08
/
30
00
:
28
:
40
tgl
Exp
$
$
Header
:
/
cvsroot
/
pgsql
/
doc
/
src
/
sgml
/
plpgsql
.
sgml
,
v
1.
6
2002
/
09
/
01
16
:
28
:
05
tgl
Exp
$
-->
-->
<
chapter
id
=
"plpgsql"
>
<
chapter
id
=
"plpgsql"
>
...
@@ -538,8 +538,6 @@ END;
...
@@ -538,8 +538,6 @@ END;
<
para
>
<
para
>
Note
that
<
literal
>
RECORD
</>
is
not
a
true
data
type
,
only
a
placeholder
.
Note
that
<
literal
>
RECORD
</>
is
not
a
true
data
type
,
only
a
placeholder
.
Thus
,
for
example
,
one
cannot
declare
a
function
returning
<
literal
>
RECORD
</>.
</
para
>
</
para
>
</
sect2
>
</
sect2
>
...
...
doc/src/sgml/xfunc.sgml
View file @
470a1048
<!--
<!--
$Header: /cvsroot/pgsql/doc/src/sgml/xfunc.sgml,v 1.
59 2002/08/30 00:28:40
tgl Exp $
$Header: /cvsroot/pgsql/doc/src/sgml/xfunc.sgml,v 1.
60 2002/09/01 16:28:05
tgl Exp $
-->
-->
<chapter id="xfunc">
<chapter id="xfunc">
...
@@ -2119,6 +2119,27 @@ SELECT * FROM vw_getfoo;
...
@@ -2119,6 +2119,27 @@ SELECT * FROM vw_getfoo;
</programlisting>
</programlisting>
are all valid statements.
are all valid statements.
</para>
</para>
<para>
In some cases it is useful to define table functions that can return
different column sets depending on how they are invoked. To support this,
the table function can be declared as returning the pseudo-type
<type>record</>. When such a function is used in a query, the expected
row structure must be specified in the query itself, so that the system
can know how to parse and plan the query. Consider this example:
<programlisting>
SELECT *
FROM dblink('dbname=template1', 'select proname, prosrc from pg_proc')
AS t1(proname name, prosrc text)
WHERE proname LIKE 'bytea%';
</programlisting>
The <literal>dblink</> function executes a remote query (see
<literal>contrib/dblink</>). It is declared to return <type>record</>
since it might be used for any kind of query. The actual column set
must be specified in the calling query so that the parser knows, for
example, what <literal>*</> should expand to.
</para>
</sect1>
</sect1>
<sect1 id="xfunc-plhandler">
<sect1 id="xfunc-plhandler">
...
...
src/pl/plpgsql/src/gram.y
View file @
470a1048
...
@@ -4,7 +4,7 @@
...
@@ -4,7 +4,7 @@
* procedural language
* procedural language
*
*
* IDENTIFICATION
* IDENTIFICATION
* $Header: /cvsroot/pgsql/src/pl/plpgsql/src/gram.y,v 1.3
6 2002/08/30 00:28:41
tgl Exp $
* $Header: /cvsroot/pgsql/src/pl/plpgsql/src/gram.y,v 1.3
7 2002/09/01 16:28:06
tgl Exp $
*
*
* This software is copyrighted by Jan Wieck - Hamburg.
* This software is copyrighted by Jan Wieck - Hamburg.
*
*
...
@@ -1159,7 +1159,6 @@ stmt_return : K_RETURN lno
...
@@ -1159,7 +1159,6 @@ stmt_return : K_RETURN lno
}
}
;
;
/* FIXME: this syntax needs work, RETURN NEXT breaks stmt_return */
stmt_return_next: K_RETURN_NEXT lno
stmt_return_next: K_RETURN_NEXT lno
{
{
PLpgSQL_stmt_return_next *new;
PLpgSQL_stmt_return_next *new;
...
...
src/pl/plpgsql/src/pl_comp.c
View file @
470a1048
...
@@ -3,7 +3,7 @@
...
@@ -3,7 +3,7 @@
* procedural language
* procedural language
*
*
* IDENTIFICATION
* IDENTIFICATION
* $Header: /cvsroot/pgsql/src/pl/plpgsql/src/pl_comp.c,v 1.
49 2002/08/30 00:28:41
tgl Exp $
* $Header: /cvsroot/pgsql/src/pl/plpgsql/src/pl_comp.c,v 1.
50 2002/09/01 16:28:06
tgl Exp $
*
*
* This software is copyrighted by Jan Wieck - Hamburg.
* This software is copyrighted by Jan Wieck - Hamburg.
*
*
...
@@ -211,11 +211,11 @@ plpgsql_compile(Oid fn_oid, int functype)
...
@@ -211,11 +211,11 @@ plpgsql_compile(Oid fn_oid, int functype)
procStruct
->
prorettype
);
procStruct
->
prorettype
);
typeStruct
=
(
Form_pg_type
)
GETSTRUCT
(
typeTup
);
typeStruct
=
(
Form_pg_type
)
GETSTRUCT
(
typeTup
);
/* Disallow pseudotype result, except VOID */
/* Disallow pseudotype result, except VOID or RECORD */
/* XXX someday allow RECORD? */
if
(
typeStruct
->
typtype
==
'p'
)
if
(
typeStruct
->
typtype
==
'p'
)
{
{
if
(
procStruct
->
prorettype
==
VOIDOID
)
if
(
procStruct
->
prorettype
==
VOIDOID
||
procStruct
->
prorettype
==
RECORDOID
)
/* okay */
;
/* okay */
;
else
if
(
procStruct
->
prorettype
==
TRIGGEROID
||
else
if
(
procStruct
->
prorettype
==
TRIGGEROID
||
procStruct
->
prorettype
==
OPAQUEOID
)
procStruct
->
prorettype
==
OPAQUEOID
)
...
@@ -227,7 +227,8 @@ plpgsql_compile(Oid fn_oid, int functype)
...
@@ -227,7 +227,8 @@ plpgsql_compile(Oid fn_oid, int functype)
format_type_be
(
procStruct
->
prorettype
));
format_type_be
(
procStruct
->
prorettype
));
}
}
if
(
typeStruct
->
typrelid
!=
InvalidOid
)
if
(
typeStruct
->
typrelid
!=
InvalidOid
||
procStruct
->
prorettype
==
RECORDOID
)
function
->
fn_retistuple
=
true
;
function
->
fn_retistuple
=
true
;
else
else
{
{
...
@@ -486,8 +487,7 @@ plpgsql_compile(Oid fn_oid, int functype)
...
@@ -486,8 +487,7 @@ plpgsql_compile(Oid fn_oid, int functype)
}
}
/*
/*
* Create the magic found variable indicating if the last FOR or
* Create the magic FOUND variable.
* SELECT statement returned data
*/
*/
var
=
malloc
(
sizeof
(
PLpgSQL_var
));
var
=
malloc
(
sizeof
(
PLpgSQL_var
));
memset
(
var
,
0
,
sizeof
(
PLpgSQL_var
));
memset
(
var
,
0
,
sizeof
(
PLpgSQL_var
));
...
...
src/pl/plpgsql/src/pl_exec.c
View file @
470a1048
...
@@ -3,7 +3,7 @@
...
@@ -3,7 +3,7 @@
* procedural language
* procedural language
*
*
* IDENTIFICATION
* IDENTIFICATION
* $Header: /cvsroot/pgsql/src/pl/plpgsql/src/pl_exec.c,v 1.6
1 2002/08/30 23:59:4
6 tgl Exp $
* $Header: /cvsroot/pgsql/src/pl/plpgsql/src/pl_exec.c,v 1.6
2 2002/09/01 16:28:0
6 tgl Exp $
*
*
* This software is copyrighted by Jan Wieck - Hamburg.
* This software is copyrighted by Jan Wieck - Hamburg.
*
*
...
@@ -1588,9 +1588,9 @@ static int
...
@@ -1588,9 +1588,9 @@ static int
exec_stmt_return_next
(
PLpgSQL_execstate
*
estate
,
exec_stmt_return_next
(
PLpgSQL_execstate
*
estate
,
PLpgSQL_stmt_return_next
*
stmt
)
PLpgSQL_stmt_return_next
*
stmt
)
{
{
TupleDesc
tupdesc
;
TupleDesc
tupdesc
;
int
natts
;
int
natts
;
HeapTuple
tuple
;
HeapTuple
tuple
;
bool
free_tuple
=
false
;
bool
free_tuple
=
false
;
if
(
!
estate
->
retisset
)
if
(
!
estate
->
retisset
)
...
...
src/test/regress/expected/plpgsql.out
View file @
470a1048
...
@@ -1680,3 +1680,56 @@ select * from test_ret_set_scalar(1,10);
...
@@ -1680,3 +1680,56 @@ select * from test_ret_set_scalar(1,10);
11
11
(10 rows)
(10 rows)
create function test_ret_set_rec_dyn(int) returns setof record as '
DECLARE
retval RECORD;
BEGIN
IF $1 > 10 THEN
SELECT INTO retval 5, 10, 15;
RETURN NEXT retval;
RETURN NEXT retval;
ELSE
SELECT INTO retval 50, 5::numeric, ''xxx''::text;
RETURN NEXT retval;
RETURN NEXT retval;
END IF;
RETURN;
END;' language 'plpgsql';
SELECT * FROM test_ret_set_rec_dyn(1500) AS (a int, b int, c int);
a | b | c
---+----+----
5 | 10 | 15
5 | 10 | 15
(2 rows)
SELECT * FROM test_ret_set_rec_dyn(5) AS (a int, b numeric, c text);
a | b | c
----+---+-----
50 | 5 | xxx
50 | 5 | xxx
(2 rows)
create function test_ret_rec_dyn(int) returns record as '
DECLARE
retval RECORD;
BEGIN
IF $1 > 10 THEN
SELECT INTO retval 5, 10, 15;
RETURN retval;
ELSE
SELECT INTO retval 50, 5::numeric, ''xxx''::text;
RETURN retval;
END IF;
END;' language 'plpgsql';
SELECT * FROM test_ret_rec_dyn(1500) AS (a int, b int, c int);
a | b | c
---+----+----
5 | 10 | 15
(1 row)
SELECT * FROM test_ret_rec_dyn(5) AS (a int, b numeric, c text);
a | b | c
----+---+-----
50 | 5 | xxx
(1 row)
src/test/regress/sql/plpgsql.sql
View file @
470a1048
...
@@ -1524,3 +1524,38 @@ BEGIN
...
@@ -1524,3 +1524,38 @@ BEGIN
END;'
language
'plpgsql'
;
END;'
language
'plpgsql'
;
select
*
from
test_ret_set_scalar
(
1
,
10
);
select
*
from
test_ret_set_scalar
(
1
,
10
);
create
function
test_ret_set_rec_dyn
(
int
)
returns
setof
record
as
'
DECLARE
retval RECORD;
BEGIN
IF $1 > 10 THEN
SELECT INTO retval 5, 10, 15;
RETURN NEXT retval;
RETURN NEXT retval;
ELSE
SELECT INTO retval 50, 5::numeric,
''
xxx
''
::text;
RETURN NEXT retval;
RETURN NEXT retval;
END IF;
RETURN;
END;'
language
'plpgsql'
;
SELECT
*
FROM
test_ret_set_rec_dyn
(
1500
)
AS
(
a
int
,
b
int
,
c
int
);
SELECT
*
FROM
test_ret_set_rec_dyn
(
5
)
AS
(
a
int
,
b
numeric
,
c
text
);
create
function
test_ret_rec_dyn
(
int
)
returns
record
as
'
DECLARE
retval RECORD;
BEGIN
IF $1 > 10 THEN
SELECT INTO retval 5, 10, 15;
RETURN retval;
ELSE
SELECT INTO retval 50, 5::numeric,
''
xxx
''
::text;
RETURN retval;
END IF;
END;'
language
'plpgsql'
;
SELECT
*
FROM
test_ret_rec_dyn
(
1500
)
AS
(
a
int
,
b
int
,
c
int
);
SELECT
*
FROM
test_ret_rec_dyn
(
5
)
AS
(
a
int
,
b
numeric
,
c
text
);
Write
Preview
Markdown
is supported
0%
Try again
or
attach a new file
Attach a file
Cancel
You are about to add
0
people
to the discussion. Proceed with caution.
Finish editing this message first!
Cancel
Please
register
or
sign in
to comment