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
819f22a3
Commit
819f22a3
authored
Sep 02, 2006
by
Bruce Momjian
Browse files
Options
Browse Files
Download
Email Patches
Plain Diff
Allow PL/python to return composite types and result sets
Sven Suursoho
parent
b1620c53
Changes
8
Expand all
Show whitespace changes
Inline
Side-by-side
Showing
8 changed files
with
1359 additions
and
99 deletions
+1359
-99
doc/src/sgml/plpython.sgml
doc/src/sgml/plpython.sgml
+287
-17
src/pl/plpython/expected/plpython_function.out
src/pl/plpython/expected/plpython_function.out
+112
-32
src/pl/plpython/expected/plpython_schema.out
src/pl/plpython/expected/plpython_schema.out
+8
-0
src/pl/plpython/expected/plpython_test.out
src/pl/plpython/expected/plpython_test.out
+341
-0
src/pl/plpython/plpython.c
src/pl/plpython/plpython.c
+407
-18
src/pl/plpython/sql/plpython_function.sql
src/pl/plpython/sql/plpython_function.sql
+124
-32
src/pl/plpython/sql/plpython_schema.sql
src/pl/plpython/sql/plpython_schema.sql
+10
-0
src/pl/plpython/sql/plpython_test.sql
src/pl/plpython/sql/plpython_test.sql
+70
-0
No files found.
doc/src/sgml/plpython.sgml
View file @
819f22a3
This diff is collapsed.
Click to expand it.
src/pl/plpython/expected/plpython_function.out
View file @
819f22a3
...
@@ -55,27 +55,27 @@ except Exception, ex:
...
@@ -55,27 +55,27 @@ except Exception, ex:
return "failed, that wasn''t supposed to happen"
return "failed, that wasn''t supposed to happen"
return "succeeded, as expected"'
return "succeeded, as expected"'
LANGUAGE plpythonu;
LANGUAGE plpythonu;
CREATE FUNCTION import_test_one(text) RETURNS text
CREATE FUNCTION import_test_one(
p
text) RETURNS text
AS
AS
'import sha
'import sha
digest = sha.new(
args[0]
)
digest = sha.new(
p
)
return digest.hexdigest()'
return digest.hexdigest()'
LANGUAGE plpythonu;
LANGUAGE plpythonu;
CREATE FUNCTION import_test_two(users) RETURNS text
CREATE FUNCTION import_test_two(u
u
sers) RETURNS text
AS
AS
'import sha
'import sha
plain =
args[0]["fname"] + args[0]
["lname"]
plain =
u["fname"] + u
["lname"]
digest = sha.new(plain);
digest = sha.new(plain);
return "sha hash of " + plain + " is " + digest.hexdigest()'
return "sha hash of " + plain + " is " + digest.hexdigest()'
LANGUAGE plpythonu;
LANGUAGE plpythonu;
CREATE FUNCTION argument_test_one(u
sers, text,
text) RETURNS text
CREATE FUNCTION argument_test_one(u
users, a1 text, a2
text) RETURNS text
AS
AS
'keys =
args[0]
.keys()
'keys =
u
.keys()
keys.sort()
keys.sort()
out = []
out = []
for key in keys:
for key in keys:
out.append("%s: %s" % (key,
args[0]
[key]))
out.append("%s: %s" % (key,
u
[key]))
words = a
rgs[1] + " " + args[2]
+ " => {" + ", ".join(out) + "}"
words = a
1 + " " + a2
+ " => {" + ", ".join(out) + "}"
return words'
return words'
LANGUAGE plpythonu;
LANGUAGE plpythonu;
-- these triggers are dedicated to HPHC of RI who
-- these triggers are dedicated to HPHC of RI who
...
@@ -174,40 +174,40 @@ DROP TRIGGER show_trigger_data_trig on trigger_test;
...
@@ -174,40 +174,40 @@ DROP TRIGGER show_trigger_data_trig on trigger_test;
DROP FUNCTION trigger_data();
DROP FUNCTION trigger_data();
-- nested calls
-- nested calls
--
--
CREATE FUNCTION nested_call_one(text) RETURNS text
CREATE FUNCTION nested_call_one(
a
text) RETURNS text
AS
AS
'q = "SELECT nested_call_two(''%s'')" % a
rgs[0]
'q = "SELECT nested_call_two(''%s'')" % a
r = plpy.execute(q)
r = plpy.execute(q)
return r[0]'
return r[0]'
LANGUAGE plpythonu ;
LANGUAGE plpythonu ;
CREATE FUNCTION nested_call_two(text) RETURNS text
CREATE FUNCTION nested_call_two(
a
text) RETURNS text
AS
AS
'q = "SELECT nested_call_three(''%s'')" % a
rgs[0]
'q = "SELECT nested_call_three(''%s'')" % a
r = plpy.execute(q)
r = plpy.execute(q)
return r[0]'
return r[0]'
LANGUAGE plpythonu ;
LANGUAGE plpythonu ;
CREATE FUNCTION nested_call_three(text) RETURNS text
CREATE FUNCTION nested_call_three(
a
text) RETURNS text
AS
AS
'return a
rgs[0]
'
'return a'
LANGUAGE plpythonu ;
LANGUAGE plpythonu ;
-- some spi stuff
-- some spi stuff
CREATE FUNCTION spi_prepared_plan_test_one(text) RETURNS text
CREATE FUNCTION spi_prepared_plan_test_one(
a
text) RETURNS text
AS
AS
'if not SD.has_key("myplan"):
'if not SD.has_key("myplan"):
q = "SELECT count(*) FROM users WHERE lname = $1"
q = "SELECT count(*) FROM users WHERE lname = $1"
SD["myplan"] = plpy.prepare(q, [ "text" ])
SD["myplan"] = plpy.prepare(q, [ "text" ])
try:
try:
rv = plpy.execute(SD["myplan"], [a
rgs[0]
])
rv = plpy.execute(SD["myplan"], [a])
return "there are " + str(rv[0]["count"]) + " " + str(a
rgs[0]
) + "s"
return "there are " + str(rv[0]["count"]) + " " + str(a) + "s"
except Exception, ex:
except Exception, ex:
plpy.error(str(ex))
plpy.error(str(ex))
return None
return None
'
'
LANGUAGE plpythonu;
LANGUAGE plpythonu;
CREATE FUNCTION spi_prepared_plan_test_nested(text) RETURNS text
CREATE FUNCTION spi_prepared_plan_test_nested(
a
text) RETURNS text
AS
AS
'if not SD.has_key("myplan"):
'if not SD.has_key("myplan"):
q = "SELECT spi_prepared_plan_test_one(''%s'') as count" % a
rgs[0]
q = "SELECT spi_prepared_plan_test_one(''%s'') as count" % a
SD["myplan"] = plpy.prepare(q)
SD["myplan"] = plpy.prepare(q)
try:
try:
rv = plpy.execute(SD["myplan"])
rv = plpy.execute(SD["myplan"])
...
@@ -223,12 +223,12 @@ return None
...
@@ -223,12 +223,12 @@ return None
CREATE FUNCTION stupid() RETURNS text AS 'return "zarkon"' LANGUAGE plpythonu;
CREATE FUNCTION stupid() RETURNS text AS 'return "zarkon"' LANGUAGE plpythonu;
/* a typo
/* a typo
*/
*/
CREATE FUNCTION invalid_type_uncaught(text) RETURNS text
CREATE FUNCTION invalid_type_uncaught(
a
text) RETURNS text
AS
AS
'if not SD.has_key("plan"):
'if not SD.has_key("plan"):
q = "SELECT fname FROM users WHERE lname = $1"
q = "SELECT fname FROM users WHERE lname = $1"
SD["plan"] = plpy.prepare(q, [ "test" ])
SD["plan"] = plpy.prepare(q, [ "test" ])
rv = plpy.execute(SD["plan"], [ a
rgs[0]
])
rv = plpy.execute(SD["plan"], [ a ])
if len(rv):
if len(rv):
return rv[0]["fname"]
return rv[0]["fname"]
return None
return None
...
@@ -237,7 +237,7 @@ return None
...
@@ -237,7 +237,7 @@ return None
/* for what it's worth catch the exception generated by
/* for what it's worth catch the exception generated by
* the typo, and return None
* the typo, and return None
*/
*/
CREATE FUNCTION invalid_type_caught(text) RETURNS text
CREATE FUNCTION invalid_type_caught(
a
text) RETURNS text
AS
AS
'if not SD.has_key("plan"):
'if not SD.has_key("plan"):
q = "SELECT fname FROM users WHERE lname = $1"
q = "SELECT fname FROM users WHERE lname = $1"
...
@@ -246,7 +246,7 @@ CREATE FUNCTION invalid_type_caught(text) RETURNS text
...
@@ -246,7 +246,7 @@ CREATE FUNCTION invalid_type_caught(text) RETURNS text
except plpy.SPIError, ex:
except plpy.SPIError, ex:
plpy.notice(str(ex))
plpy.notice(str(ex))
return None
return None
rv = plpy.execute(SD["plan"], [ a
rgs[0]
])
rv = plpy.execute(SD["plan"], [ a ])
if len(rv):
if len(rv):
return rv[0]["fname"]
return rv[0]["fname"]
return None
return None
...
@@ -255,7 +255,7 @@ return None
...
@@ -255,7 +255,7 @@ return None
/* for what it's worth catch the exception generated by
/* for what it's worth catch the exception generated by
* the typo, and reraise it as a plain error
* the typo, and reraise it as a plain error
*/
*/
CREATE FUNCTION invalid_type_reraised(text) RETURNS text
CREATE FUNCTION invalid_type_reraised(
a
text) RETURNS text
AS
AS
'if not SD.has_key("plan"):
'if not SD.has_key("plan"):
q = "SELECT fname FROM users WHERE lname = $1"
q = "SELECT fname FROM users WHERE lname = $1"
...
@@ -263,7 +263,7 @@ CREATE FUNCTION invalid_type_reraised(text) RETURNS text
...
@@ -263,7 +263,7 @@ CREATE FUNCTION invalid_type_reraised(text) RETURNS text
SD["plan"] = plpy.prepare(q, [ "test" ])
SD["plan"] = plpy.prepare(q, [ "test" ])
except plpy.SPIError, ex:
except plpy.SPIError, ex:
plpy.error(str(ex))
plpy.error(str(ex))
rv = plpy.execute(SD["plan"], [ a
rgs[0]
])
rv = plpy.execute(SD["plan"], [ a ])
if len(rv):
if len(rv):
return rv[0]["fname"]
return rv[0]["fname"]
return None
return None
...
@@ -271,11 +271,11 @@ return None
...
@@ -271,11 +271,11 @@ return None
LANGUAGE plpythonu;
LANGUAGE plpythonu;
/* no typo no messing about
/* no typo no messing about
*/
*/
CREATE FUNCTION valid_type(text) RETURNS text
CREATE FUNCTION valid_type(
a
text) RETURNS text
AS
AS
'if not SD.has_key("plan"):
'if not SD.has_key("plan"):
SD["plan"] = plpy.prepare("SELECT fname FROM users WHERE lname = $1", [ "text" ])
SD["plan"] = plpy.prepare("SELECT fname FROM users WHERE lname = $1", [ "text" ])
rv = plpy.execute(SD["plan"], [ a
rgs[0]
])
rv = plpy.execute(SD["plan"], [ a ])
if len(rv):
if len(rv):
return rv[0]["fname"]
return rv[0]["fname"]
return None
return None
...
@@ -300,13 +300,13 @@ CREATE FUNCTION exception_index_invalid_nested() RETURNS text
...
@@ -300,13 +300,13 @@ CREATE FUNCTION exception_index_invalid_nested() RETURNS text
'rv = plpy.execute("SELECT test5(''foo'')")
'rv = plpy.execute("SELECT test5(''foo'')")
return rv[0]'
return rv[0]'
LANGUAGE plpythonu;
LANGUAGE plpythonu;
CREATE FUNCTION join_sequences(sequences) RETURNS text
CREATE FUNCTION join_sequences(s
s
equences) RETURNS text
AS
AS
'if not
args[0]
["multipart"]:
'if not
s
["multipart"]:
return
args[0]
["sequence"]
return
s
["sequence"]
q = "SELECT sequence FROM xsequences WHERE pid = ''%s''" %
args[0]
["pid"]
q = "SELECT sequence FROM xsequences WHERE pid = ''%s''" %
s
["pid"]
rv = plpy.execute(q)
rv = plpy.execute(q)
seq =
args[0]
["sequence"]
seq =
s
["sequence"]
for r in rv:
for r in rv:
seq = seq + r["sequence"]
seq = seq + r["sequence"]
return seq
return seq
...
@@ -357,3 +357,83 @@ $$ LANGUAGE plpythonu;
...
@@ -357,3 +357,83 @@ $$ LANGUAGE plpythonu;
CREATE FUNCTION test_return_none() RETURNS int AS $$
CREATE FUNCTION test_return_none() RETURNS int AS $$
None
None
$$ LANGUAGE plpythonu;
$$ LANGUAGE plpythonu;
--
-- Test named parameters
--
CREATE FUNCTION test_param_names1(a0 integer, a1 text) RETURNS boolean AS $$
assert a0 == args[0]
assert a1 == args[1]
return True
$$ LANGUAGE plpythonu;
CREATE FUNCTION test_param_names2(u users) RETURNS text AS $$
assert u == args[0]
return str(u)
$$ LANGUAGE plpythonu;
-- use deliberately wrong parameter names
CREATE FUNCTION test_param_names3(a0 integer) RETURNS boolean AS $$
try:
assert a1 == args[0]
return False
except NameError, e:
assert e.args[0].find("a1") > -1
return True
$$ LANGUAGE plpythonu;
--
-- Test returning SETOF
--
CREATE FUNCTION test_setof_as_list(count integer, content text) RETURNS SETOF text AS $$
return [ content ]*count
$$ LANGUAGE plpythonu;
CREATE FUNCTION test_setof_as_tuple(count integer, content text) RETURNS SETOF text AS $$
t = ()
for i in xrange(count):
t += ( content, )
return t
$$ LANGUAGE plpythonu;
CREATE FUNCTION test_setof_as_iterator(count integer, content text) RETURNS SETOF text AS $$
class producer:
def __init__ (self, icount, icontent):
self.icontent = icontent
self.icount = icount
def __iter__ (self):
return self
def next (self):
if self.icount == 0:
raise StopIteration
self.icount -= 1
return self.icontent
return producer(count, content)
$$ LANGUAGE plpythonu;
--
-- Test returning tuples
--
CREATE FUNCTION test_table_record_as(typ text, first text, second integer, retnull boolean) RETURNS table_record AS $$
if retnull:
return None
if typ == 'dict':
return { 'first': first, 'second': second, 'additionalfield': 'must not cause trouble' }
elif typ == 'tuple':
return ( first, second )
elif typ == 'list':
return [ first, second ]
elif typ == 'obj':
class type_record: pass
type_record.first = first
type_record.second = second
return type_record
$$ LANGUAGE plpythonu;
CREATE FUNCTION test_type_record_as(typ text, first text, second integer, retnull boolean) RETURNS type_record AS $$
if retnull:
return None
if typ == 'dict':
return { 'first': first, 'second': second, 'additionalfield': 'must not cause trouble' }
elif typ == 'tuple':
return ( first, second )
elif typ == 'list':
return [ first, second ]
elif typ == 'obj':
class type_record: pass
type_record.first = first
type_record.second = second
return type_record
$$ LANGUAGE plpythonu;
src/pl/plpython/expected/plpython_schema.out
View file @
819f22a3
...
@@ -44,3 +44,11 @@ CREATE INDEX xsequences_pid_idx ON xsequences(pid) ;
...
@@ -44,3 +44,11 @@ CREATE INDEX xsequences_pid_idx ON xsequences(pid) ;
CREATE TABLE unicode_test (
CREATE TABLE unicode_test (
testvalue text NOT NULL
testvalue text NOT NULL
);
);
CREATE TABLE table_record (
first text,
second int4
) ;
CREATE TYPE type_record AS (
first text,
second int4
) ;
src/pl/plpython/expected/plpython_test.out
View file @
819f22a3
...
@@ -198,3 +198,344 @@ SELECT test_return_none(), test_return_none() IS NULL AS "is null";
...
@@ -198,3 +198,344 @@ SELECT test_return_none(), test_return_none() IS NULL AS "is null";
| t
| t
(1 row)
(1 row)
-- Test for functions with named parameters
SELECT test_param_names1(1,'text');
test_param_names1
-------------------
t
(1 row)
SELECT test_param_names2(users) from users;
test_param_names2
----------------------------------------------------------------------------
{'lname': 'doe', 'username': 'j_doe', 'userid': 1, 'fname': 'jane'}
{'lname': 'doe', 'username': 'johnd', 'userid': 2, 'fname': 'john'}
{'lname': 'doe', 'username': 'w_doe', 'userid': 3, 'fname': 'willem'}
{'lname': 'smith', 'username': 'slash', 'userid': 4, 'fname': 'rick'}
{'lname': 'smith', 'username': 'w_smith', 'userid': 5, 'fname': 'willem'}
{'lname': 'darwin', 'username': 'beagle', 'userid': 6, 'fname': 'charles'}
(6 rows)
SELECT test_param_names3(1);
test_param_names3
-------------------
t
(1 row)
-- Test set returning functions
SELECT test_setof_as_list(0, 'list');
test_setof_as_list
--------------------
(0 rows)
SELECT test_setof_as_list(1, 'list');
test_setof_as_list
--------------------
list
(1 row)
SELECT test_setof_as_list(2, 'list');
test_setof_as_list
--------------------
list
list
(2 rows)
SELECT test_setof_as_list(2, null);
test_setof_as_list
--------------------
(2 rows)
SELECT test_setof_as_tuple(0, 'tuple');
test_setof_as_tuple
---------------------
(0 rows)
SELECT test_setof_as_tuple(1, 'tuple');
test_setof_as_tuple
---------------------
tuple
(1 row)
SELECT test_setof_as_tuple(2, 'tuple');
test_setof_as_tuple
---------------------
tuple
tuple
(2 rows)
SELECT test_setof_as_tuple(2, null);
test_setof_as_tuple
---------------------
(2 rows)
SELECT test_setof_as_iterator(0, 'list');
test_setof_as_iterator
------------------------
(0 rows)
SELECT test_setof_as_iterator(1, 'list');
test_setof_as_iterator
------------------------
list
(1 row)
SELECT test_setof_as_iterator(2, 'list');
test_setof_as_iterator
------------------------
list
list
(2 rows)
SELECT test_setof_as_iterator(2, null);
test_setof_as_iterator
------------------------
(2 rows)
-- Test tuple returning functions
SELECT * FROM test_table_record_as('dict', null, null, false);
first | second
-------+--------
|
(1 row)
SELECT * FROM test_table_record_as('dict', 'one', null, false);
first | second
-------+--------
one |
(1 row)
SELECT * FROM test_table_record_as('dict', null, 2, false);
first | second
-------+--------
| 2
(1 row)
SELECT * FROM test_table_record_as('dict', 'three', 3, false);
first | second
-------+--------
three | 3
(1 row)
SELECT * FROM test_table_record_as('dict', null, null, true);
first | second
-------+--------
|
(1 row)
SELECT * FROM test_table_record_as('tuple', null, null, false);
first | second
-------+--------
|
(1 row)
SELECT * FROM test_table_record_as('tuple', 'one', null, false);
first | second
-------+--------
one |
(1 row)
SELECT * FROM test_table_record_as('tuple', null, 2, false);
first | second
-------+--------
| 2
(1 row)
SELECT * FROM test_table_record_as('tuple', 'three', 3, false);
first | second
-------+--------
three | 3
(1 row)
SELECT * FROM test_table_record_as('tuple', null, null, true);
first | second
-------+--------
|
(1 row)
SELECT * FROM test_table_record_as('list', null, null, false);
first | second
-------+--------
|
(1 row)
SELECT * FROM test_table_record_as('list', 'one', null, false);
first | second
-------+--------
one |
(1 row)
SELECT * FROM test_table_record_as('list', null, 2, false);
first | second
-------+--------
| 2
(1 row)
SELECT * FROM test_table_record_as('list', 'three', 3, false);
first | second
-------+--------
three | 3
(1 row)
SELECT * FROM test_table_record_as('list', null, null, true);
first | second
-------+--------
|
(1 row)
SELECT * FROM test_table_record_as('obj', null, null, false);
first | second
-------+--------
|
(1 row)
SELECT * FROM test_table_record_as('obj', 'one', null, false);
first | second
-------+--------
one |
(1 row)
SELECT * FROM test_table_record_as('obj', null, 2, false);
first | second
-------+--------
| 2
(1 row)
SELECT * FROM test_table_record_as('obj', 'three', 3, false);
first | second
-------+--------
three | 3
(1 row)
SELECT * FROM test_table_record_as('obj', null, null, true);
first | second
-------+--------
|
(1 row)
SELECT * FROM test_type_record_as('dict', null, null, false);
first | second
-------+--------
|
(1 row)
SELECT * FROM test_type_record_as('dict', 'one', null, false);
first | second
-------+--------
one |
(1 row)
SELECT * FROM test_type_record_as('dict', null, 2, false);
first | second
-------+--------
| 2
(1 row)
SELECT * FROM test_type_record_as('dict', 'three', 3, false);
first | second
-------+--------
three | 3
(1 row)
SELECT * FROM test_type_record_as('dict', null, null, true);
first | second
-------+--------
|
(1 row)
SELECT * FROM test_type_record_as('tuple', null, null, false);
first | second
-------+--------
|
(1 row)
SELECT * FROM test_type_record_as('tuple', 'one', null, false);
first | second
-------+--------
one |
(1 row)
SELECT * FROM test_type_record_as('tuple', null, 2, false);
first | second
-------+--------
| 2
(1 row)
SELECT * FROM test_type_record_as('tuple', 'three', 3, false);
first | second
-------+--------
three | 3
(1 row)
SELECT * FROM test_type_record_as('tuple', null, null, true);
first | second
-------+--------
|
(1 row)
SELECT * FROM test_type_record_as('list', null, null, false);
first | second
-------+--------
|
(1 row)
SELECT * FROM test_type_record_as('list', 'one', null, false);
first | second
-------+--------
one |
(1 row)
SELECT * FROM test_type_record_as('list', null, 2, false);
first | second
-------+--------
| 2
(1 row)
SELECT * FROM test_type_record_as('list', 'three', 3, false);
first | second
-------+--------
three | 3
(1 row)
SELECT * FROM test_type_record_as('list', null, null, true);
first | second
-------+--------
|
(1 row)
SELECT * FROM test_type_record_as('obj', null, null, false);
first | second
-------+--------
|
(1 row)
SELECT * FROM test_type_record_as('obj', 'one', null, false);
first | second
-------+--------
one |
(1 row)
SELECT * FROM test_type_record_as('obj', null, 2, false);
first | second
-------+--------
| 2
(1 row)
SELECT * FROM test_type_record_as('obj', 'three', 3, false);
first | second
-------+--------
three | 3
(1 row)
SELECT * FROM test_type_record_as('obj', null, null, true);
first | second
-------+--------
|
(1 row)
src/pl/plpython/plpython.c
View file @
819f22a3
This diff is collapsed.
Click to expand it.
src/pl/plpython/sql/plpython_function.sql
View file @
819f22a3
...
@@ -65,29 +65,29 @@ except Exception, ex:
...
@@ -65,29 +65,29 @@ except Exception, ex:
return "succeeded, as expected"'
return "succeeded, as expected"'
LANGUAGE
plpythonu
;
LANGUAGE
plpythonu
;
CREATE
FUNCTION
import_test_one
(
text
)
RETURNS
text
CREATE
FUNCTION
import_test_one
(
p
text
)
RETURNS
text
AS
AS
'import sha
'import sha
digest = sha.new(
args[0]
)
digest = sha.new(
p
)
return digest.hexdigest()'
return digest.hexdigest()'
LANGUAGE
plpythonu
;
LANGUAGE
plpythonu
;
CREATE
FUNCTION
import_test_two
(
users
)
RETURNS
text
CREATE
FUNCTION
import_test_two
(
u
u
sers
)
RETURNS
text
AS
AS
'import sha
'import sha
plain =
args[0]["fname"] + args[0]
["lname"]
plain =
u["fname"] + u
["lname"]
digest = sha.new(plain);
digest = sha.new(plain);
return "sha hash of " + plain + " is " + digest.hexdigest()'
return "sha hash of " + plain + " is " + digest.hexdigest()'
LANGUAGE
plpythonu
;
LANGUAGE
plpythonu
;
CREATE
FUNCTION
argument_test_one
(
u
sers
,
text
,
text
)
RETURNS
text
CREATE
FUNCTION
argument_test_one
(
u
users
,
a1
text
,
a2
text
)
RETURNS
text
AS
AS
'keys =
args[0]
.keys()
'keys =
u
.keys()
keys.sort()
keys.sort()
out = []
out = []
for key in keys:
for key in keys:
out.append("%s: %s" % (key,
args[0]
[key]))
out.append("%s: %s" % (key,
u
[key]))
words = a
rgs[1] + " " + args[2]
+ " => {" + ", ".join(out) + "}"
words = a
1 + " " + a2
+ " => {" + ", ".join(out) + "}"
return words'
return words'
LANGUAGE
plpythonu
;
LANGUAGE
plpythonu
;
...
@@ -176,45 +176,45 @@ DROP FUNCTION trigger_data();
...
@@ -176,45 +176,45 @@ DROP FUNCTION trigger_data();
-- nested calls
-- nested calls
--
--
CREATE
FUNCTION
nested_call_one
(
text
)
RETURNS
text
CREATE
FUNCTION
nested_call_one
(
a
text
)
RETURNS
text
AS
AS
'q = "SELECT nested_call_two(
''
%s
''
)" % a
rgs[0]
'q = "SELECT nested_call_two(
''
%s
''
)" % a
r = plpy.execute(q)
r = plpy.execute(q)
return r[0]'
return r[0]'
LANGUAGE
plpythonu
;
LANGUAGE
plpythonu
;
CREATE
FUNCTION
nested_call_two
(
text
)
RETURNS
text
CREATE
FUNCTION
nested_call_two
(
a
text
)
RETURNS
text
AS
AS
'q = "SELECT nested_call_three(
''
%s
''
)" % a
rgs[0]
'q = "SELECT nested_call_three(
''
%s
''
)" % a
r = plpy.execute(q)
r = plpy.execute(q)
return r[0]'
return r[0]'
LANGUAGE
plpythonu
;
LANGUAGE
plpythonu
;
CREATE
FUNCTION
nested_call_three
(
text
)
RETURNS
text
CREATE
FUNCTION
nested_call_three
(
a
text
)
RETURNS
text
AS
AS
'return a
rgs[0]
'
'return a'
LANGUAGE
plpythonu
;
LANGUAGE
plpythonu
;
-- some spi stuff
-- some spi stuff
CREATE
FUNCTION
spi_prepared_plan_test_one
(
text
)
RETURNS
text
CREATE
FUNCTION
spi_prepared_plan_test_one
(
a
text
)
RETURNS
text
AS
AS
'if not SD.has_key("myplan"):
'if not SD.has_key("myplan"):
q = "SELECT count(*) FROM users WHERE lname = $1"
q = "SELECT count(*) FROM users WHERE lname = $1"
SD["myplan"] = plpy.prepare(q, [ "text" ])
SD["myplan"] = plpy.prepare(q, [ "text" ])
try:
try:
rv = plpy.execute(SD["myplan"], [a
rgs[0]
])
rv = plpy.execute(SD["myplan"], [a])
return "there are " + str(rv[0]["count"]) + " " + str(a
rgs[0]
) + "s"
return "there are " + str(rv[0]["count"]) + " " + str(a) + "s"
except Exception, ex:
except Exception, ex:
plpy.error(str(ex))
plpy.error(str(ex))
return None
return None
'
'
LANGUAGE
plpythonu
;
LANGUAGE
plpythonu
;
CREATE
FUNCTION
spi_prepared_plan_test_nested
(
text
)
RETURNS
text
CREATE
FUNCTION
spi_prepared_plan_test_nested
(
a
text
)
RETURNS
text
AS
AS
'if not SD.has_key("myplan"):
'if not SD.has_key("myplan"):
q = "SELECT spi_prepared_plan_test_one(
''
%s
''
) as count" % a
rgs[0]
q = "SELECT spi_prepared_plan_test_one(
''
%s
''
) as count" % a
SD["myplan"] = plpy.prepare(q)
SD["myplan"] = plpy.prepare(q)
try:
try:
rv = plpy.execute(SD["myplan"])
rv = plpy.execute(SD["myplan"])
...
@@ -233,12 +233,12 @@ CREATE FUNCTION stupid() RETURNS text AS 'return "zarkon"' LANGUAGE plpythonu;
...
@@ -233,12 +233,12 @@ CREATE FUNCTION stupid() RETURNS text AS 'return "zarkon"' LANGUAGE plpythonu;
/* a typo
/* a typo
*/
*/
CREATE
FUNCTION
invalid_type_uncaught
(
text
)
RETURNS
text
CREATE
FUNCTION
invalid_type_uncaught
(
a
text
)
RETURNS
text
AS
AS
'if not SD.has_key("plan"):
'if not SD.has_key("plan"):
q = "SELECT fname FROM users WHERE lname = $1"
q = "SELECT fname FROM users WHERE lname = $1"
SD["plan"] = plpy.prepare(q, [ "test" ])
SD["plan"] = plpy.prepare(q, [ "test" ])
rv = plpy.execute(SD["plan"], [ a
rgs[0]
])
rv = plpy.execute(SD["plan"], [ a ])
if len(rv):
if len(rv):
return rv[0]["fname"]
return rv[0]["fname"]
return None
return None
...
@@ -248,7 +248,7 @@ return None
...
@@ -248,7 +248,7 @@ return None
/* for what it's worth catch the exception generated by
/* for what it's worth catch the exception generated by
* the typo, and return None
* the typo, and return None
*/
*/
CREATE
FUNCTION
invalid_type_caught
(
text
)
RETURNS
text
CREATE
FUNCTION
invalid_type_caught
(
a
text
)
RETURNS
text
AS
AS
'if not SD.has_key("plan"):
'if not SD.has_key("plan"):
q = "SELECT fname FROM users WHERE lname = $1"
q = "SELECT fname FROM users WHERE lname = $1"
...
@@ -257,7 +257,7 @@ CREATE FUNCTION invalid_type_caught(text) RETURNS text
...
@@ -257,7 +257,7 @@ CREATE FUNCTION invalid_type_caught(text) RETURNS text
except plpy.SPIError, ex:
except plpy.SPIError, ex:
plpy.notice(str(ex))
plpy.notice(str(ex))
return None
return None
rv = plpy.execute(SD["plan"], [ a
rgs[0]
])
rv = plpy.execute(SD["plan"], [ a ])
if len(rv):
if len(rv):
return rv[0]["fname"]
return rv[0]["fname"]
return None
return None
...
@@ -267,7 +267,7 @@ return None
...
@@ -267,7 +267,7 @@ return None
/* for what it's worth catch the exception generated by
/* for what it's worth catch the exception generated by
* the typo, and reraise it as a plain error
* the typo, and reraise it as a plain error
*/
*/
CREATE
FUNCTION
invalid_type_reraised
(
text
)
RETURNS
text
CREATE
FUNCTION
invalid_type_reraised
(
a
text
)
RETURNS
text
AS
AS
'if not SD.has_key("plan"):
'if not SD.has_key("plan"):
q = "SELECT fname FROM users WHERE lname = $1"
q = "SELECT fname FROM users WHERE lname = $1"
...
@@ -275,7 +275,7 @@ CREATE FUNCTION invalid_type_reraised(text) RETURNS text
...
@@ -275,7 +275,7 @@ CREATE FUNCTION invalid_type_reraised(text) RETURNS text
SD["plan"] = plpy.prepare(q, [ "test" ])
SD["plan"] = plpy.prepare(q, [ "test" ])
except plpy.SPIError, ex:
except plpy.SPIError, ex:
plpy.error(str(ex))
plpy.error(str(ex))
rv = plpy.execute(SD["plan"], [ a
rgs[0]
])
rv = plpy.execute(SD["plan"], [ a ])
if len(rv):
if len(rv):
return rv[0]["fname"]
return rv[0]["fname"]
return None
return None
...
@@ -285,11 +285,11 @@ return None
...
@@ -285,11 +285,11 @@ return None
/* no typo no messing about
/* no typo no messing about
*/
*/
CREATE
FUNCTION
valid_type
(
text
)
RETURNS
text
CREATE
FUNCTION
valid_type
(
a
text
)
RETURNS
text
AS
AS
'if not SD.has_key("plan"):
'if not SD.has_key("plan"):
SD["plan"] = plpy.prepare("SELECT fname FROM users WHERE lname = $1", [ "text" ])
SD["plan"] = plpy.prepare("SELECT fname FROM users WHERE lname = $1", [ "text" ])
rv = plpy.execute(SD["plan"], [ a
rgs[0]
])
rv = plpy.execute(SD["plan"], [ a ])
if len(rv):
if len(rv):
return rv[0]["fname"]
return rv[0]["fname"]
return None
return None
...
@@ -318,13 +318,13 @@ return rv[0]'
...
@@ -318,13 +318,13 @@ return rv[0]'
LANGUAGE
plpythonu
;
LANGUAGE
plpythonu
;
CREATE
FUNCTION
join_sequences
(
sequences
)
RETURNS
text
CREATE
FUNCTION
join_sequences
(
s
s
equences
)
RETURNS
text
AS
AS
'if not
args[0]
["multipart"]:
'if not
s
["multipart"]:
return
args[0]
["sequence"]
return
s
["sequence"]
q = "SELECT sequence FROM xsequences WHERE pid =
''
%s
''
" %
args[0]
["pid"]
q = "SELECT sequence FROM xsequences WHERE pid =
''
%s
''
" %
s
["pid"]
rv = plpy.execute(q)
rv = plpy.execute(q)
seq =
args[0]
["sequence"]
seq =
s
["sequence"]
for r in rv:
for r in rv:
seq = seq + r["sequence"]
seq = seq + r["sequence"]
return seq
return seq
...
@@ -389,3 +389,95 @@ $$ LANGUAGE plpythonu;
...
@@ -389,3 +389,95 @@ $$ LANGUAGE plpythonu;
CREATE
FUNCTION
test_return_none
()
RETURNS
int
AS
$$
CREATE
FUNCTION
test_return_none
()
RETURNS
int
AS
$$
None
None
$$
LANGUAGE
plpythonu
;
$$
LANGUAGE
plpythonu
;
--
-- Test named parameters
--
CREATE
FUNCTION
test_param_names1
(
a0
integer
,
a1
text
)
RETURNS
boolean
AS
$$
assert
a0
==
args
[
0
]
assert
a1
==
args
[
1
]
return
True
$$
LANGUAGE
plpythonu
;
CREATE
FUNCTION
test_param_names2
(
u
users
)
RETURNS
text
AS
$$
assert
u
==
args
[
0
]
return
str
(
u
)
$$
LANGUAGE
plpythonu
;
-- use deliberately wrong parameter names
CREATE
FUNCTION
test_param_names3
(
a0
integer
)
RETURNS
boolean
AS
$$
try
:
assert
a1
==
args
[
0
]
return
False
except
NameError
,
e
:
assert
e
.
args
[
0
].
find
(
"a1"
)
>
-
1
return
True
$$
LANGUAGE
plpythonu
;
--
-- Test returning SETOF
--
CREATE
FUNCTION
test_setof_as_list
(
count
integer
,
content
text
)
RETURNS
SETOF
text
AS
$$
return
[
content
]
*
count
$$
LANGUAGE
plpythonu
;
CREATE
FUNCTION
test_setof_as_tuple
(
count
integer
,
content
text
)
RETURNS
SETOF
text
AS
$$
t
=
()
for
i
in
xrange
(
count
):
t
+=
(
content
,
)
return
t
$$
LANGUAGE
plpythonu
;
CREATE
FUNCTION
test_setof_as_iterator
(
count
integer
,
content
text
)
RETURNS
SETOF
text
AS
$$
class
producer
:
def
__init__
(
self
,
icount
,
icontent
):
self
.
icontent
=
icontent
self
.
icount
=
icount
def
__iter__
(
self
):
return
self
def
next
(
self
):
if
self
.
icount
==
0
:
raise
StopIteration
self
.
icount
-=
1
return
self
.
icontent
return
producer
(
count
,
content
)
$$
LANGUAGE
plpythonu
;
--
-- Test returning tuples
--
CREATE
FUNCTION
test_table_record_as
(
typ
text
,
first
text
,
second
integer
,
retnull
boolean
)
RETURNS
table_record
AS
$$
if
retnull
:
return
None
if
typ
==
'dict'
:
return
{
'first'
:
first
,
'second'
:
second
,
'additionalfield'
:
'must not cause trouble'
}
elif
typ
==
'tuple'
:
return
(
first
,
second
)
elif
typ
==
'list'
:
return
[
first
,
second
]
elif
typ
==
'obj'
:
class
type_record
:
pass
type_record
.
first
=
first
type_record
.
second
=
second
return
type_record
$$
LANGUAGE
plpythonu
;
CREATE
FUNCTION
test_type_record_as
(
typ
text
,
first
text
,
second
integer
,
retnull
boolean
)
RETURNS
type_record
AS
$$
if
retnull
:
return
None
if
typ
==
'dict'
:
return
{
'first'
:
first
,
'second'
:
second
,
'additionalfield'
:
'must not cause trouble'
}
elif
typ
==
'tuple'
:
return
(
first
,
second
)
elif
typ
==
'list'
:
return
[
first
,
second
]
elif
typ
==
'obj'
:
class
type_record
:
pass
type_record
.
first
=
first
type_record
.
second
=
second
return
type_record
$$
LANGUAGE
plpythonu
;
src/pl/plpython/sql/plpython_schema.sql
View file @
819f22a3
...
@@ -42,3 +42,13 @@ CREATE INDEX xsequences_pid_idx ON xsequences(pid) ;
...
@@ -42,3 +42,13 @@ CREATE INDEX xsequences_pid_idx ON xsequences(pid) ;
CREATE
TABLE
unicode_test
(
CREATE
TABLE
unicode_test
(
testvalue
text
NOT
NULL
testvalue
text
NOT
NULL
);
);
CREATE
TABLE
table_record
(
first
text
,
second
int4
)
;
CREATE
TYPE
type_record
AS
(
first
text
,
second
int4
)
;
src/pl/plpython/sql/plpython_test.sql
View file @
819f22a3
...
@@ -73,3 +73,73 @@ SELECT newline_crlf();
...
@@ -73,3 +73,73 @@ SELECT newline_crlf();
SELECT
test_void_func1
(),
test_void_func1
()
IS
NULL
AS
"is null"
;
SELECT
test_void_func1
(),
test_void_func1
()
IS
NULL
AS
"is null"
;
SELECT
test_void_func2
();
-- should fail
SELECT
test_void_func2
();
-- should fail
SELECT
test_return_none
(),
test_return_none
()
IS
NULL
AS
"is null"
;
SELECT
test_return_none
(),
test_return_none
()
IS
NULL
AS
"is null"
;
-- Test for functions with named parameters
SELECT
test_param_names1
(
1
,
'text'
);
SELECT
test_param_names2
(
users
)
from
users
;
SELECT
test_param_names3
(
1
);
-- Test set returning functions
SELECT
test_setof_as_list
(
0
,
'list'
);
SELECT
test_setof_as_list
(
1
,
'list'
);
SELECT
test_setof_as_list
(
2
,
'list'
);
SELECT
test_setof_as_list
(
2
,
null
);
SELECT
test_setof_as_tuple
(
0
,
'tuple'
);
SELECT
test_setof_as_tuple
(
1
,
'tuple'
);
SELECT
test_setof_as_tuple
(
2
,
'tuple'
);
SELECT
test_setof_as_tuple
(
2
,
null
);
SELECT
test_setof_as_iterator
(
0
,
'list'
);
SELECT
test_setof_as_iterator
(
1
,
'list'
);
SELECT
test_setof_as_iterator
(
2
,
'list'
);
SELECT
test_setof_as_iterator
(
2
,
null
);
-- Test tuple returning functions
SELECT
*
FROM
test_table_record_as
(
'dict'
,
null
,
null
,
false
);
SELECT
*
FROM
test_table_record_as
(
'dict'
,
'one'
,
null
,
false
);
SELECT
*
FROM
test_table_record_as
(
'dict'
,
null
,
2
,
false
);
SELECT
*
FROM
test_table_record_as
(
'dict'
,
'three'
,
3
,
false
);
SELECT
*
FROM
test_table_record_as
(
'dict'
,
null
,
null
,
true
);
SELECT
*
FROM
test_table_record_as
(
'tuple'
,
null
,
null
,
false
);
SELECT
*
FROM
test_table_record_as
(
'tuple'
,
'one'
,
null
,
false
);
SELECT
*
FROM
test_table_record_as
(
'tuple'
,
null
,
2
,
false
);
SELECT
*
FROM
test_table_record_as
(
'tuple'
,
'three'
,
3
,
false
);
SELECT
*
FROM
test_table_record_as
(
'tuple'
,
null
,
null
,
true
);
SELECT
*
FROM
test_table_record_as
(
'list'
,
null
,
null
,
false
);
SELECT
*
FROM
test_table_record_as
(
'list'
,
'one'
,
null
,
false
);
SELECT
*
FROM
test_table_record_as
(
'list'
,
null
,
2
,
false
);
SELECT
*
FROM
test_table_record_as
(
'list'
,
'three'
,
3
,
false
);
SELECT
*
FROM
test_table_record_as
(
'list'
,
null
,
null
,
true
);
SELECT
*
FROM
test_table_record_as
(
'obj'
,
null
,
null
,
false
);
SELECT
*
FROM
test_table_record_as
(
'obj'
,
'one'
,
null
,
false
);
SELECT
*
FROM
test_table_record_as
(
'obj'
,
null
,
2
,
false
);
SELECT
*
FROM
test_table_record_as
(
'obj'
,
'three'
,
3
,
false
);
SELECT
*
FROM
test_table_record_as
(
'obj'
,
null
,
null
,
true
);
SELECT
*
FROM
test_type_record_as
(
'dict'
,
null
,
null
,
false
);
SELECT
*
FROM
test_type_record_as
(
'dict'
,
'one'
,
null
,
false
);
SELECT
*
FROM
test_type_record_as
(
'dict'
,
null
,
2
,
false
);
SELECT
*
FROM
test_type_record_as
(
'dict'
,
'three'
,
3
,
false
);
SELECT
*
FROM
test_type_record_as
(
'dict'
,
null
,
null
,
true
);
SELECT
*
FROM
test_type_record_as
(
'tuple'
,
null
,
null
,
false
);
SELECT
*
FROM
test_type_record_as
(
'tuple'
,
'one'
,
null
,
false
);
SELECT
*
FROM
test_type_record_as
(
'tuple'
,
null
,
2
,
false
);
SELECT
*
FROM
test_type_record_as
(
'tuple'
,
'three'
,
3
,
false
);
SELECT
*
FROM
test_type_record_as
(
'tuple'
,
null
,
null
,
true
);
SELECT
*
FROM
test_type_record_as
(
'list'
,
null
,
null
,
false
);
SELECT
*
FROM
test_type_record_as
(
'list'
,
'one'
,
null
,
false
);
SELECT
*
FROM
test_type_record_as
(
'list'
,
null
,
2
,
false
);
SELECT
*
FROM
test_type_record_as
(
'list'
,
'three'
,
3
,
false
);
SELECT
*
FROM
test_type_record_as
(
'list'
,
null
,
null
,
true
);
SELECT
*
FROM
test_type_record_as
(
'obj'
,
null
,
null
,
false
);
SELECT
*
FROM
test_type_record_as
(
'obj'
,
'one'
,
null
,
false
);
SELECT
*
FROM
test_type_record_as
(
'obj'
,
null
,
2
,
false
);
SELECT
*
FROM
test_type_record_as
(
'obj'
,
'three'
,
3
,
false
);
SELECT
*
FROM
test_type_record_as
(
'obj'
,
null
,
null
,
true
);
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