Commit 6a1e2b3c authored by Joe Conway's avatar Joe Conway

Added new versions of dblink, dblink_exec, dblink_open, dblink_close,

and, dblink_fetch -- allows ERROR on remote side of connection to
throw NOTICE locally instead of ERROR. Also removed documentation for
previously deprecated, now removed, functions.
parent cb659ecb
...@@ -30,13 +30,11 @@ ...@@ -30,13 +30,11 @@
* *
*/ */
Version 0.6 (14 June, 2003):
Completely removed previously deprecated functions. Added ability
to create "named" persistent connections in addition to the single global
"unnamed" persistent connection.
Tested under Linux (Red Hat 9) and PostgreSQL 7.4devel.
Release Notes: Release Notes:
Version 0.7 (as of 25 Feb, 2004)
- Added new version of dblink, dblink_exec, dblink_open, dblink_close,
and, dblink_fetch -- allows ERROR on remote side of connection to
throw NOTICE locally instead of ERROR
Version 0.6 Version 0.6
- functions deprecated in 0.5 have been removed - functions deprecated in 0.5 have been removed
- added ability to create "named" persistent connections - added ability to create "named" persistent connections
...@@ -85,7 +83,7 @@ Installation: ...@@ -85,7 +83,7 @@ Installation:
You can use dblink.sql to create the functions in your database of choice, e.g. You can use dblink.sql to create the functions in your database of choice, e.g.
psql -U postgres template1 < dblink.sql psql template1 < dblink.sql
installs following functions into database template1: installs following functions into database template1:
...@@ -104,40 +102,40 @@ Installation: ...@@ -104,40 +102,40 @@ Installation:
cursor cursor
------------ ------------
dblink_open(text,text) RETURNS text dblink_open(text,text [, bool fail_on_error]) RETURNS text
- opens a cursor using unnamed connection already opened with - opens a cursor using unnamed connection already opened with
dblink_connect() that will persist for duration of current backend dblink_connect() that will persist for duration of current backend
or until it is closed or until it is closed
dblink_open(text,text,text) RETURNS text dblink_open(text,text,text [, bool fail_on_error]) RETURNS text
- opens a cursor using a named connection already opened with - opens a cursor using a named connection already opened with
dblink_connect() that will persist for duration of current backend dblink_connect() that will persist for duration of current backend
or until it is closed or until it is closed
dblink_fetch(text, int) RETURNS setof record dblink_fetch(text, int [, bool fail_on_error]) RETURNS setof record
- fetches data from an already opened cursor on the unnamed connection - fetches data from an already opened cursor on the unnamed connection
dblink_fetch(text, text, int) RETURNS setof record dblink_fetch(text, text, int [, bool fail_on_error]) RETURNS setof record
- fetches data from an already opened cursor on a named connection - fetches data from an already opened cursor on a named connection
dblink_close(text) RETURNS text dblink_close(text [, bool fail_on_error]) RETURNS text
- closes a cursor on the unnamed connection - closes a cursor on the unnamed connection
dblink_close(text,text) RETURNS text dblink_close(text,text [, bool fail_on_error]) RETURNS text
- closes a cursor on a named connection - closes a cursor on a named connection
query query
------------ ------------
dblink(text,text) RETURNS setof record dblink(text,text [, bool fail_on_error]) RETURNS setof record
- returns a set of results from remote SELECT query; the first argument - returns a set of results from remote SELECT query; the first argument
is either a connection string, or the name of an already opened is either a connection string, or the name of an already opened
persistant connection persistant connection
dblink(text) RETURNS setof record dblink(text [, bool fail_on_error]) RETURNS setof record
- returns a set of results from remote SELECT query, using the unnamed - returns a set of results from remote SELECT query, using the unnamed
connection already opened with dblink_connect() connection already opened with dblink_connect()
execute execute
------------ ------------
dblink_exec(text, text) RETURNS text dblink_exec(text, text [, bool fail_on_error]) RETURNS text
- executes an INSERT/UPDATE/DELETE query remotely; the first argument - executes an INSERT/UPDATE/DELETE query remotely; the first argument
is either a connection string, or the name of an already opened is either a connection string, or the name of an already opened
persistant connection persistant connection
dblink_exec(text) RETURNS text dblink_exec(text [, bool fail_on_error]) RETURNS text
- executes an INSERT/UPDATE/DELETE query remotely, using connection - executes an INSERT/UPDATE/DELETE query remotely, using connection
already opened with dblink_connect() already opened with dblink_connect()
...@@ -169,7 +167,6 @@ Documentation: ...@@ -169,7 +167,6 @@ Documentation:
doc/query doc/query
doc/execute doc/execute
doc/misc doc/misc
doc/deprecated
================================================================== ==================================================================
-- Joe Conway -- Joe Conway
......
This diff is collapsed.
CREATE OR REPLACE FUNCTION dblink_connect (text) CREATE OR REPLACE FUNCTION dblink_connect (text)
RETURNS text RETURNS text
AS 'MODULE_PATHNAME','dblink_connect' AS 'MODULE_PATHNAME','dblink_connect'
LANGUAGE 'C' WITH (isstrict); LANGUAGE 'C' STRICT;
CREATE OR REPLACE FUNCTION dblink_connect (text, text) CREATE OR REPLACE FUNCTION dblink_connect (text, text)
RETURNS text RETURNS text
AS 'MODULE_PATHNAME','dblink_connect' AS 'MODULE_PATHNAME','dblink_connect'
LANGUAGE 'C' WITH (isstrict); LANGUAGE 'C' STRICT;
CREATE OR REPLACE FUNCTION dblink_disconnect () CREATE OR REPLACE FUNCTION dblink_disconnect ()
RETURNS text RETURNS text
AS 'MODULE_PATHNAME','dblink_disconnect' AS 'MODULE_PATHNAME','dblink_disconnect'
LANGUAGE 'C' WITH (isstrict); LANGUAGE 'C' STRICT;
CREATE OR REPLACE FUNCTION dblink_disconnect (text) CREATE OR REPLACE FUNCTION dblink_disconnect (text)
RETURNS text RETURNS text
AS 'MODULE_PATHNAME','dblink_disconnect' AS 'MODULE_PATHNAME','dblink_disconnect'
LANGUAGE 'C' WITH (isstrict); LANGUAGE 'C' STRICT;
CREATE OR REPLACE FUNCTION dblink_open (text,text) CREATE OR REPLACE FUNCTION dblink_open (text,text)
RETURNS text RETURNS text
AS 'MODULE_PATHNAME','dblink_open' AS 'MODULE_PATHNAME','dblink_open'
LANGUAGE 'C' WITH (isstrict); LANGUAGE 'C' STRICT;
CREATE OR REPLACE FUNCTION dblink_open (text,text,bool)
RETURNS text
AS 'MODULE_PATHNAME','dblink_open'
LANGUAGE 'C' STRICT;
CREATE OR REPLACE FUNCTION dblink_open (text,text,text) CREATE OR REPLACE FUNCTION dblink_open (text,text,text)
RETURNS text RETURNS text
AS 'MODULE_PATHNAME','dblink_open' AS 'MODULE_PATHNAME','dblink_open'
LANGUAGE 'C' WITH (isstrict); LANGUAGE 'C' STRICT;
CREATE OR REPLACE FUNCTION dblink_open (text,text,text,bool)
RETURNS text
AS 'MODULE_PATHNAME','dblink_open'
LANGUAGE 'C' STRICT;
CREATE OR REPLACE FUNCTION dblink_fetch (text,int) CREATE OR REPLACE FUNCTION dblink_fetch (text,int)
RETURNS setof record RETURNS setof record
AS 'MODULE_PATHNAME','dblink_fetch' AS 'MODULE_PATHNAME','dblink_fetch'
LANGUAGE 'C' WITH (isstrict); LANGUAGE 'C' STRICT;
CREATE OR REPLACE FUNCTION dblink_fetch (text,int,bool)
RETURNS setof record
AS 'MODULE_PATHNAME','dblink_fetch'
LANGUAGE 'C' STRICT;
CREATE OR REPLACE FUNCTION dblink_fetch (text,text,int) CREATE OR REPLACE FUNCTION dblink_fetch (text,text,int)
RETURNS setof record RETURNS setof record
AS 'MODULE_PATHNAME','dblink_fetch' AS 'MODULE_PATHNAME','dblink_fetch'
LANGUAGE 'C' WITH (isstrict); LANGUAGE 'C' STRICT;
CREATE OR REPLACE FUNCTION dblink_fetch (text,text,int,bool)
RETURNS setof record
AS 'MODULE_PATHNAME','dblink_fetch'
LANGUAGE 'C' STRICT;
CREATE OR REPLACE FUNCTION dblink_close (text) CREATE OR REPLACE FUNCTION dblink_close (text)
RETURNS text RETURNS text
AS 'MODULE_PATHNAME','dblink_close' AS 'MODULE_PATHNAME','dblink_close'
LANGUAGE 'C' WITH (isstrict); LANGUAGE 'C' STRICT;
CREATE OR REPLACE FUNCTION dblink_close (text,bool)
RETURNS text
AS 'MODULE_PATHNAME','dblink_close'
LANGUAGE 'C' STRICT;
CREATE OR REPLACE FUNCTION dblink_close (text,text) CREATE OR REPLACE FUNCTION dblink_close (text,text)
RETURNS text RETURNS text
AS 'MODULE_PATHNAME','dblink_close' AS 'MODULE_PATHNAME','dblink_close'
LANGUAGE 'C' WITH (isstrict); LANGUAGE 'C' STRICT;
CREATE OR REPLACE FUNCTION dblink_close (text,text,bool)
RETURNS text
AS 'MODULE_PATHNAME','dblink_close'
LANGUAGE 'C' STRICT;
CREATE OR REPLACE FUNCTION dblink (text,text) CREATE OR REPLACE FUNCTION dblink (text,text)
RETURNS setof record RETURNS setof record
AS 'MODULE_PATHNAME','dblink_record' AS 'MODULE_PATHNAME','dblink_record'
LANGUAGE 'C' WITH (isstrict); LANGUAGE 'C' STRICT;
CREATE OR REPLACE FUNCTION dblink (text,text,bool)
RETURNS setof record
AS 'MODULE_PATHNAME','dblink_record'
LANGUAGE 'C' STRICT;
CREATE OR REPLACE FUNCTION dblink (text) CREATE OR REPLACE FUNCTION dblink (text)
RETURNS setof record RETURNS setof record
AS 'MODULE_PATHNAME','dblink_record' AS 'MODULE_PATHNAME','dblink_record'
LANGUAGE 'C' WITH (isstrict); LANGUAGE 'C' STRICT;
CREATE OR REPLACE FUNCTION dblink (text,bool)
RETURNS setof record
AS 'MODULE_PATHNAME','dblink_record'
LANGUAGE 'C' STRICT;
CREATE OR REPLACE FUNCTION dblink_exec (text,text) CREATE OR REPLACE FUNCTION dblink_exec (text,text)
RETURNS text RETURNS text
AS 'MODULE_PATHNAME','dblink_exec' AS 'MODULE_PATHNAME','dblink_exec'
LANGUAGE 'C' WITH (isstrict); LANGUAGE 'C' STRICT;
CREATE OR REPLACE FUNCTION dblink_exec (text,text,bool)
RETURNS text
AS 'MODULE_PATHNAME','dblink_exec'
LANGUAGE 'C' STRICT;
CREATE OR REPLACE FUNCTION dblink_exec (text) CREATE OR REPLACE FUNCTION dblink_exec (text)
RETURNS text RETURNS text
AS 'MODULE_PATHNAME','dblink_exec' AS 'MODULE_PATHNAME','dblink_exec'
LANGUAGE 'c' WITH (isstrict); LANGUAGE 'c' STRICT;
CREATE OR REPLACE FUNCTION dblink_exec (text,bool)
RETURNS text
AS 'MODULE_PATHNAME','dblink_exec'
LANGUAGE 'c' STRICT;
CREATE TYPE dblink_pkey_results AS (position int4, colname text); CREATE TYPE dblink_pkey_results AS (position int4, colname text);
CREATE OR REPLACE FUNCTION dblink_get_pkey (text) CREATE OR REPLACE FUNCTION dblink_get_pkey (text)
RETURNS setof dblink_pkey_results RETURNS setof dblink_pkey_results
AS 'MODULE_PATHNAME','dblink_get_pkey' AS 'MODULE_PATHNAME','dblink_get_pkey'
LANGUAGE 'c' WITH (isstrict); LANGUAGE 'c' STRICT;
CREATE OR REPLACE FUNCTION dblink_build_sql_insert (text, int2vector, int4, _text, _text) CREATE OR REPLACE FUNCTION dblink_build_sql_insert (text, int2vector, int4, _text, _text)
RETURNS text RETURNS text
AS 'MODULE_PATHNAME','dblink_build_sql_insert' AS 'MODULE_PATHNAME','dblink_build_sql_insert'
LANGUAGE 'C' WITH (isstrict); LANGUAGE 'C' STRICT;
CREATE OR REPLACE FUNCTION dblink_build_sql_delete (text, int2vector, int4, _text) CREATE OR REPLACE FUNCTION dblink_build_sql_delete (text, int2vector, int4, _text)
RETURNS text RETURNS text
AS 'MODULE_PATHNAME','dblink_build_sql_delete' AS 'MODULE_PATHNAME','dblink_build_sql_delete'
LANGUAGE 'C' WITH (isstrict); LANGUAGE 'C' STRICT;
CREATE OR REPLACE FUNCTION dblink_build_sql_update (text, int2vector, int4, _text, _text) CREATE OR REPLACE FUNCTION dblink_build_sql_update (text, int2vector, int4, _text, _text)
RETURNS text RETURNS text
AS 'MODULE_PATHNAME','dblink_build_sql_update' AS 'MODULE_PATHNAME','dblink_build_sql_update'
LANGUAGE 'C' WITH (isstrict); LANGUAGE 'C' STRICT;
CREATE OR REPLACE FUNCTION dblink_current_query () CREATE OR REPLACE FUNCTION dblink_current_query ()
RETURNS text RETURNS text
......
...@@ -5,8 +5,8 @@ dblink_open -- Opens a cursor on a remote database ...@@ -5,8 +5,8 @@ dblink_open -- Opens a cursor on a remote database
Synopsis Synopsis
dblink_open(text cursorname, text sql) dblink_open(text cursorname, text sql [, bool fail_on_error])
dblink_open(text connname, text cursorname, text sql) dblink_open(text connname, text cursorname, text sql [, bool fail_on_error])
Inputs Inputs
...@@ -23,6 +23,13 @@ Inputs ...@@ -23,6 +23,13 @@ Inputs
sql statement that you wish to execute on the remote host sql statement that you wish to execute on the remote host
e.g. "select * from pg_class" e.g. "select * from pg_class"
fail_on_error
If true (default when not present) then an ERROR thrown on the remote side
of the connection causes an ERROR to also be thrown locally. If false, the
remote ERROR is locally treated as a NOTICE, and the return value is set
to 'ERROR'.
Outputs Outputs
Returns status = "OK" Returns status = "OK"
...@@ -56,8 +63,8 @@ dblink_fetch -- Returns a set from an open cursor on a remote database ...@@ -56,8 +63,8 @@ dblink_fetch -- Returns a set from an open cursor on a remote database
Synopsis Synopsis
dblink_fetch(text cursorname, int32 howmany) dblink_fetch(text cursorname, int32 howmany [, bool fail_on_error])
dblink_fetch(text connname, text cursorname, int32 howmany) dblink_fetch(text connname, text cursorname, int32 howmany [, bool fail_on_error])
Inputs Inputs
...@@ -75,6 +82,12 @@ Inputs ...@@ -75,6 +82,12 @@ Inputs
starting at the current cursor position, moving forward. Once the cursor starting at the current cursor position, moving forward. Once the cursor
has positioned to the end, no more rows are produced. has positioned to the end, no more rows are produced.
fail_on_error
If true (default when not present) then an ERROR thrown on the remote side
of the connection causes an ERROR to also be thrown locally. If false, the
remote ERROR is locally treated as a NOTICE, and no rows are returned.
Outputs Outputs
Returns setof record Returns setof record
...@@ -132,8 +145,8 @@ dblink_close -- Closes a cursor on a remote database ...@@ -132,8 +145,8 @@ dblink_close -- Closes a cursor on a remote database
Synopsis Synopsis
dblink_close(text cursorname) dblink_close(text cursorname [, bool fail_on_error])
dblink_close(text connname, text cursorname) dblink_close(text connname, text cursorname [, bool fail_on_error])
Inputs Inputs
...@@ -145,6 +158,13 @@ Inputs ...@@ -145,6 +158,13 @@ Inputs
a reference name for the cursor a reference name for the cursor
fail_on_error
If true (default when not present) then an ERROR thrown on the remote side
of the connection causes an ERROR to also be thrown locally. If false, the
remote ERROR is locally treated as a NOTICE, and the return value is set
to 'ERROR'.
Outputs Outputs
Returns status = "OK" Returns status = "OK"
......
==================================================================
Name
*DEPRECATED* use new dblink syntax
dblink -- Returns a resource id for a data set from a remote database
Synopsis
dblink(text connstr, text sql)
Inputs
connstr
standard libpq format connection srting,
e.g. "hostaddr=127.0.0.1 port=5432 dbname=mydb user=postgres password=mypasswd"
sql
sql statement that you wish to execute on the remote host
e.g. "select * from pg_class"
Outputs
Returns setof int (res_id)
Example usage
select dblink('hostaddr=127.0.0.1 port=5432 dbname=mydb user=postgres password=mypasswd'
,'select f1, f2 from mytable');
==================================================================
Name
*DEPRECATED* use new dblink syntax
dblink_tok -- Returns individual select field results from a dblink remote query
Synopsis
dblink_tok(int res_id, int fnumber)
Inputs
res_id
a resource id returned by a call to dblink()
fnumber
the ordinal position (zero based) of the field to be returned from the dblink result set
Outputs
Returns text
Example usage
select dblink_tok(t1.dblink_p,0) as f1, dblink_tok(t1.dblink_p,1) as f2
from (select dblink('hostaddr=127.0.0.1 port=5432 dbname=mydb user=postgres password=mypasswd'
,'select f1, f2 from mytable') as dblink_p) as t1;
==================================================================
*DEPRECATED* use new dblink syntax
A more convenient way to use dblink may be to create a view:
create view myremotetable as
select dblink_tok(t1.dblink_p,0) as f1, dblink_tok(t1.dblink_p,1) as f2
from (select dblink('hostaddr=127.0.0.1 port=5432 dbname=template1 user=postgres password=postgres'
,'select proname, prosrc from pg_proc') as dblink_p) as t1;
Then you can simply write:
select f1, f2 from myremotetable where f1 like 'bytea%';
==================================================================
Name
*DEPRECATED* use new dblink_exec syntax
dblink_last_oid -- Returns last inserted oid
Synopsis
dblink_last_oid(int res_id) RETURNS oid
Inputs
res_id
any resource id returned by dblink function;
Outputs
Returns oid of last inserted tuple
Example usage
test=# select dblink_last_oid(dblink('hostaddr=127.0.0.1 port=5432 dbname=mydb user=postgres password=mypasswd'
,'insert into mytable (f1, f2) values (1,2)'));
dblink_last_oid
----------------
16553
(1 row)
...@@ -5,14 +5,15 @@ dblink_exec -- Executes an UPDATE/INSERT/DELETE on a remote database ...@@ -5,14 +5,15 @@ dblink_exec -- Executes an UPDATE/INSERT/DELETE on a remote database
Synopsis Synopsis
dblink_exec(text connstr, text sql) dblink_exec(text connstr, text sql [, bool fail_on_error])
dblink_exec(text connname, text sql) dblink_exec(text connname, text sql [, bool fail_on_error])
dblink_exec(text sql) dblink_exec(text sql [, bool fail_on_error])
Inputs Inputs
connname connname
connstr connstr
If two arguments are present, the first is first assumed to be a specific If two arguments are present, the first is first assumed to be a specific
connection name to use. If the name is not found, the argument is then connection name to use. If the name is not found, the argument is then
assumed to be a valid connection string, of standard libpq format, assumed to be a valid connection string, of standard libpq format,
...@@ -25,9 +26,16 @@ Inputs ...@@ -25,9 +26,16 @@ Inputs
sql statement that you wish to execute on the remote host, e.g.: sql statement that you wish to execute on the remote host, e.g.:
insert into foo values(0,'a','{"a0","b0","c0"}'); insert into foo values(0,'a','{"a0","b0","c0"}');
fail_on_error
If true (default when not present) then an ERROR thrown on the remote side
of the connection causes an ERROR to also be thrown locally. If false, the
remote ERROR is locally treated as a NOTICE, and the return value is set
to 'ERROR'.
Outputs Outputs
Returns status of the command Returns status of the command, or 'ERROR' if the command failed.
Notes Notes
1) dblink_open starts an explicit transaction. If, after using dblink_open, 1) dblink_open starts an explicit transaction. If, after using dblink_open,
...@@ -60,3 +68,12 @@ select dblink_exec('myconn','insert into foo values(21,''z'',''{"a0","b0","c0"}' ...@@ -60,3 +68,12 @@ select dblink_exec('myconn','insert into foo values(21,''z'',''{"a0","b0","c0"}'
------------------ ------------------
INSERT 6432584 1 INSERT 6432584 1
(1 row) (1 row)
select dblink_exec('myconn','insert into pg_class values (''foo'')',false);
NOTICE: sql error
DETAIL: ERROR: null value in column "relnamespace" violates not-null constraint
dblink_exec
-------------
ERROR
(1 row)
...@@ -5,9 +5,9 @@ dblink -- Returns a set from a remote database ...@@ -5,9 +5,9 @@ dblink -- Returns a set from a remote database
Synopsis Synopsis
dblink(text connstr, text sql) dblink(text connstr, text sql [, bool fail_on_error])
dblink(text connname, text sql) dblink(text connname, text sql [, bool fail_on_error])
dblink(text sql) dblink(text sql [, bool fail_on_error])
Inputs Inputs
...@@ -25,6 +25,12 @@ Inputs ...@@ -25,6 +25,12 @@ Inputs
sql statement that you wish to execute on the remote host sql statement that you wish to execute on the remote host
e.g. "select * from pg_class" e.g. "select * from pg_class"
fail_on_error
If true (default when not present) then an ERROR thrown on the remote side
of the connection causes an ERROR to also be thrown locally. If false, the
remote ERROR is locally treated as a NOTICE, and no rows are returned.
Outputs Outputs
Returns setof record Returns setof record
......
...@@ -128,6 +128,23 @@ WHERE t.a > 7; ...@@ -128,6 +128,23 @@ WHERE t.a > 7;
9 | j | {a9,b9,c9} 9 | j | {a9,b9,c9}
(2 rows) (2 rows)
-- open a cursor with bad SQL and fail_on_error set to false
SELECT dblink_open('rmt_foo_cursor','SELECT * FROM foobar',false);
NOTICE: sql error
DETAIL: ERROR: relation "foobar" does not exist
dblink_open
-------------
ERROR
(1 row)
-- reset remote transaction state
SELECT dblink_exec('ABORT');
dblink_exec
-------------
ROLLBACK
(1 row)
-- open a cursor -- open a cursor
SELECT dblink_open('rmt_foo_cursor','SELECT * FROM foo'); SELECT dblink_open('rmt_foo_cursor','SELECT * FROM foo');
dblink_open dblink_open
...@@ -135,6 +152,20 @@ SELECT dblink_open('rmt_foo_cursor','SELECT * FROM foo'); ...@@ -135,6 +152,20 @@ SELECT dblink_open('rmt_foo_cursor','SELECT * FROM foo');
OK OK
(1 row) (1 row)
-- close the cursor
SELECT dblink_close('rmt_foo_cursor',false);
dblink_close
--------------
OK
(1 row)
-- open the cursor again
SELECT dblink_open('rmt_foo_cursor','SELECT * FROM foo');
dblink_open
-------------
OK
(1 row)
-- fetch some data -- fetch some data
SELECT * SELECT *
FROM dblink_fetch('rmt_foo_cursor',4) AS t(a int, b text, c text[]); FROM dblink_fetch('rmt_foo_cursor',4) AS t(a int, b text, c text[]);
...@@ -165,11 +196,38 @@ FROM dblink_fetch('rmt_foo_cursor',4) AS t(a int, b text, c text[]); ...@@ -165,11 +196,38 @@ FROM dblink_fetch('rmt_foo_cursor',4) AS t(a int, b text, c text[]);
9 | j | {a9,b9,c9} 9 | j | {a9,b9,c9}
(2 rows) (2 rows)
-- close the cursor -- intentionally botch a fetch
SELECT dblink_close('rmt_foo_cursor'); SELECT *
FROM dblink_fetch('rmt_foobar_cursor',4,false) AS t(a int, b text, c text[]);
NOTICE: sql error
DETAIL: ERROR: cursor "rmt_foobar_cursor" does not exist
a | b | c
---+---+---
(0 rows)
-- reset remote transaction state
SELECT dblink_exec('ABORT');
dblink_exec
-------------
ROLLBACK
(1 row)
-- close the wrong cursor
SELECT dblink_close('rmt_foobar_cursor',false);
NOTICE: sql error
DETAIL: ERROR: cursor "rmt_foobar_cursor" does not exist
dblink_close dblink_close
-------------- --------------
OK ERROR
(1 row)
-- reset remote transaction state
SELECT dblink_exec('ABORT');
dblink_exec
-------------
ROLLBACK
(1 row) (1 row)
-- should generate 'cursor "rmt_foo_cursor" not found' error -- should generate 'cursor "rmt_foo_cursor" not found' error
...@@ -178,6 +236,16 @@ FROM dblink_fetch('rmt_foo_cursor',4) AS t(a int, b text, c text[]); ...@@ -178,6 +236,16 @@ FROM dblink_fetch('rmt_foo_cursor',4) AS t(a int, b text, c text[]);
ERROR: sql error ERROR: sql error
DETAIL: ERROR: cursor "rmt_foo_cursor" does not exist DETAIL: ERROR: cursor "rmt_foo_cursor" does not exist
-- this time, 'cursor "rmt_foo_cursor" not found' as a notice
SELECT *
FROM dblink_fetch('rmt_foo_cursor',4,false) AS t(a int, b text, c text[]);
NOTICE: sql error
DETAIL: ERROR: cursor "rmt_foo_cursor" does not exist
a | b | c
---+---+---
(0 rows)
-- close the persistent connection -- close the persistent connection
SELECT dblink_disconnect(); SELECT dblink_disconnect();
dblink_disconnect dblink_disconnect
...@@ -232,6 +300,23 @@ FROM dblink('SELECT * FROM foo') AS t(a int, b text, c text[]); ...@@ -232,6 +300,23 @@ FROM dblink('SELECT * FROM foo') AS t(a int, b text, c text[]);
11 | l | {a11,b11,c11} 11 | l | {a11,b11,c11}
(12 rows) (12 rows)
-- bad remote select
SELECT *
FROM dblink('SELECT * FROM foobar',false) AS t(a int, b text, c text[]);
NOTICE: sql error
DETAIL: ERROR: relation "foobar" does not exist
a | b | c
---+---+---
(0 rows)
-- reset remote transaction state
SELECT dblink_exec('ABORT');
dblink_exec
-------------
ROLLBACK
(1 row)
-- change some data -- change some data
SELECT dblink_exec('UPDATE foo SET f3[2] = ''b99'' WHERE f1 = 11'); SELECT dblink_exec('UPDATE foo SET f3[2] = ''b99'' WHERE f1 = 11');
dblink_exec dblink_exec
...@@ -248,6 +333,23 @@ WHERE a = 11; ...@@ -248,6 +333,23 @@ WHERE a = 11;
11 | l | {a11,b99,c11} 11 | l | {a11,b99,c11}
(1 row) (1 row)
-- botch a change to some other data
SELECT dblink_exec('UPDATE foobar SET f3[2] = ''b99'' WHERE f1 = 11',false);
NOTICE: sql error
DETAIL: ERROR: relation "foobar" does not exist
dblink_exec
-------------
ERROR
(1 row)
-- reset remote transaction state
SELECT dblink_exec('ABORT');
dblink_exec
-------------
ROLLBACK
(1 row)
-- delete some data -- delete some data
SELECT dblink_exec('DELETE FROM foo WHERE f1 = 11'); SELECT dblink_exec('DELETE FROM foo WHERE f1 = 11');
dblink_exec dblink_exec
...@@ -298,6 +400,24 @@ WHERE t.a > 7; ...@@ -298,6 +400,24 @@ WHERE t.a > 7;
10 | k | {a10,b10,c10} 10 | k | {a10,b10,c10}
(3 rows) (3 rows)
-- use the named persistent connection, but get it wrong
SELECT *
FROM dblink('myconn','SELECT * FROM foobar',false) AS t(a int, b text, c text[])
WHERE t.a > 7;
NOTICE: sql error
DETAIL: ERROR: relation "foobar" does not exist
a | b | c
---+---+---
(0 rows)
-- reset remote transaction state
SELECT dblink_exec('myconn','ABORT');
dblink_exec
-------------
ROLLBACK
(1 row)
-- create a second named persistent connection -- create a second named persistent connection
-- should error with "duplicate connection name" -- should error with "duplicate connection name"
SELECT dblink_connect('myconn','dbname=regression'); SELECT dblink_connect('myconn','dbname=regression');
...@@ -327,6 +447,23 @@ SELECT dblink_disconnect('myconn2'); ...@@ -327,6 +447,23 @@ SELECT dblink_disconnect('myconn2');
OK OK
(1 row) (1 row)
-- open a cursor incorrectly
SELECT dblink_open('myconn','rmt_foo_cursor','SELECT * FROM foobar',false);
NOTICE: sql error
DETAIL: ERROR: relation "foobar" does not exist
dblink_open
-------------
ERROR
(1 row)
-- reset remote transaction state
SELECT dblink_exec('myconn','ABORT');
dblink_exec
-------------
ROLLBACK
(1 row)
-- open a cursor -- open a cursor
SELECT dblink_open('myconn','rmt_foo_cursor','SELECT * FROM foo'); SELECT dblink_open('myconn','rmt_foo_cursor','SELECT * FROM foo');
dblink_open dblink_open
...@@ -365,11 +502,21 @@ FROM dblink_fetch('myconn','rmt_foo_cursor',4) AS t(a int, b text, c text[]); ...@@ -365,11 +502,21 @@ FROM dblink_fetch('myconn','rmt_foo_cursor',4) AS t(a int, b text, c text[]);
10 | k | {a10,b10,c10} 10 | k | {a10,b10,c10}
(3 rows) (3 rows)
-- close the cursor -- fetch some data incorrectly
SELECT dblink_close('myconn','rmt_foo_cursor'); SELECT *
dblink_close FROM dblink_fetch('myconn','rmt_foobar_cursor',4,false) AS t(a int, b text, c text[]);
-------------- NOTICE: sql error
OK DETAIL: ERROR: cursor "rmt_foobar_cursor" does not exist
a | b | c
---+---+---
(0 rows)
-- reset remote transaction state
SELECT dblink_exec('myconn','ABORT');
dblink_exec
-------------
ROLLBACK
(1 row) (1 row)
-- should generate 'cursor "rmt_foo_cursor" not found' error -- should generate 'cursor "rmt_foo_cursor" not found' error
......
...@@ -81,9 +81,21 @@ SELECT * ...@@ -81,9 +81,21 @@ SELECT *
FROM dblink('SELECT * FROM foo') AS t(a int, b text, c text[]) FROM dblink('SELECT * FROM foo') AS t(a int, b text, c text[])
WHERE t.a > 7; WHERE t.a > 7;
-- open a cursor with bad SQL and fail_on_error set to false
SELECT dblink_open('rmt_foo_cursor','SELECT * FROM foobar',false);
-- reset remote transaction state
SELECT dblink_exec('ABORT');
-- open a cursor -- open a cursor
SELECT dblink_open('rmt_foo_cursor','SELECT * FROM foo'); SELECT dblink_open('rmt_foo_cursor','SELECT * FROM foo');
-- close the cursor
SELECT dblink_close('rmt_foo_cursor',false);
-- open the cursor again
SELECT dblink_open('rmt_foo_cursor','SELECT * FROM foo');
-- fetch some data -- fetch some data
SELECT * SELECT *
FROM dblink_fetch('rmt_foo_cursor',4) AS t(a int, b text, c text[]); FROM dblink_fetch('rmt_foo_cursor',4) AS t(a int, b text, c text[]);
...@@ -95,13 +107,27 @@ FROM dblink_fetch('rmt_foo_cursor',4) AS t(a int, b text, c text[]); ...@@ -95,13 +107,27 @@ FROM dblink_fetch('rmt_foo_cursor',4) AS t(a int, b text, c text[]);
SELECT * SELECT *
FROM dblink_fetch('rmt_foo_cursor',4) AS t(a int, b text, c text[]); FROM dblink_fetch('rmt_foo_cursor',4) AS t(a int, b text, c text[]);
-- close the cursor -- intentionally botch a fetch
SELECT dblink_close('rmt_foo_cursor'); SELECT *
FROM dblink_fetch('rmt_foobar_cursor',4,false) AS t(a int, b text, c text[]);
-- reset remote transaction state
SELECT dblink_exec('ABORT');
-- close the wrong cursor
SELECT dblink_close('rmt_foobar_cursor',false);
-- reset remote transaction state
SELECT dblink_exec('ABORT');
-- should generate 'cursor "rmt_foo_cursor" not found' error -- should generate 'cursor "rmt_foo_cursor" not found' error
SELECT * SELECT *
FROM dblink_fetch('rmt_foo_cursor',4) AS t(a int, b text, c text[]); FROM dblink_fetch('rmt_foo_cursor',4) AS t(a int, b text, c text[]);
-- this time, 'cursor "rmt_foo_cursor" not found' as a notice
SELECT *
FROM dblink_fetch('rmt_foo_cursor',4,false) AS t(a int, b text, c text[]);
-- close the persistent connection -- close the persistent connection
SELECT dblink_disconnect(); SELECT dblink_disconnect();
...@@ -125,6 +151,13 @@ SELECT substr(dblink_exec('INSERT INTO foo VALUES(11,''l'',''{"a11","b11","c11"} ...@@ -125,6 +151,13 @@ SELECT substr(dblink_exec('INSERT INTO foo VALUES(11,''l'',''{"a11","b11","c11"}
SELECT * SELECT *
FROM dblink('SELECT * FROM foo') AS t(a int, b text, c text[]); FROM dblink('SELECT * FROM foo') AS t(a int, b text, c text[]);
-- bad remote select
SELECT *
FROM dblink('SELECT * FROM foobar',false) AS t(a int, b text, c text[]);
-- reset remote transaction state
SELECT dblink_exec('ABORT');
-- change some data -- change some data
SELECT dblink_exec('UPDATE foo SET f3[2] = ''b99'' WHERE f1 = 11'); SELECT dblink_exec('UPDATE foo SET f3[2] = ''b99'' WHERE f1 = 11');
...@@ -133,6 +166,12 @@ SELECT * ...@@ -133,6 +166,12 @@ SELECT *
FROM dblink('SELECT * FROM foo') AS t(a int, b text, c text[]) FROM dblink('SELECT * FROM foo') AS t(a int, b text, c text[])
WHERE a = 11; WHERE a = 11;
-- botch a change to some other data
SELECT dblink_exec('UPDATE foobar SET f3[2] = ''b99'' WHERE f1 = 11',false);
-- reset remote transaction state
SELECT dblink_exec('ABORT');
-- delete some data -- delete some data
SELECT dblink_exec('DELETE FROM foo WHERE f1 = 11'); SELECT dblink_exec('DELETE FROM foo WHERE f1 = 11');
...@@ -161,6 +200,14 @@ SELECT * ...@@ -161,6 +200,14 @@ SELECT *
FROM dblink('myconn','SELECT * FROM foo') AS t(a int, b text, c text[]) FROM dblink('myconn','SELECT * FROM foo') AS t(a int, b text, c text[])
WHERE t.a > 7; WHERE t.a > 7;
-- use the named persistent connection, but get it wrong
SELECT *
FROM dblink('myconn','SELECT * FROM foobar',false) AS t(a int, b text, c text[])
WHERE t.a > 7;
-- reset remote transaction state
SELECT dblink_exec('myconn','ABORT');
-- create a second named persistent connection -- create a second named persistent connection
-- should error with "duplicate connection name" -- should error with "duplicate connection name"
SELECT dblink_connect('myconn','dbname=regression'); SELECT dblink_connect('myconn','dbname=regression');
...@@ -176,6 +223,12 @@ WHERE t.a > 7; ...@@ -176,6 +223,12 @@ WHERE t.a > 7;
-- close the second named persistent connection -- close the second named persistent connection
SELECT dblink_disconnect('myconn2'); SELECT dblink_disconnect('myconn2');
-- open a cursor incorrectly
SELECT dblink_open('myconn','rmt_foo_cursor','SELECT * FROM foobar',false);
-- reset remote transaction state
SELECT dblink_exec('myconn','ABORT');
-- open a cursor -- open a cursor
SELECT dblink_open('myconn','rmt_foo_cursor','SELECT * FROM foo'); SELECT dblink_open('myconn','rmt_foo_cursor','SELECT * FROM foo');
...@@ -190,8 +243,12 @@ FROM dblink_fetch('myconn','rmt_foo_cursor',4) AS t(a int, b text, c text[]); ...@@ -190,8 +243,12 @@ FROM dblink_fetch('myconn','rmt_foo_cursor',4) AS t(a int, b text, c text[]);
SELECT * SELECT *
FROM dblink_fetch('myconn','rmt_foo_cursor',4) AS t(a int, b text, c text[]); FROM dblink_fetch('myconn','rmt_foo_cursor',4) AS t(a int, b text, c text[]);
-- close the cursor -- fetch some data incorrectly
SELECT dblink_close('myconn','rmt_foo_cursor'); SELECT *
FROM dblink_fetch('myconn','rmt_foobar_cursor',4,false) AS t(a int, b text, c text[]);
-- reset remote transaction state
SELECT dblink_exec('myconn','ABORT');
-- should generate 'cursor "rmt_foo_cursor" not found' error -- should generate 'cursor "rmt_foo_cursor" not found' error
SELECT * SELECT *
......
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