46.3. Встроенные функции
46.3.1. Обращение к базе данных из PL/Perl
Обращаться к самой базе данных из кода Perl можно, используя следующие функции:
-
spi_exec_query(query[,предел]) Функция
spi_exec_queryвыполняет команду SQL и возвращает весь набор строк в виде ссылки на массив хешей. Еслипределзадан и больше нуля, тоspi_exec_queryполучает число строк, не превышающеепредел, как если бы запрос включал предложениеLIMIT. Без указанияпределаили когда он равен нулю, ограничение на количество строк снимается.Эту функцию следует использовать, только если вы знаете, что набор будет относительно небольшим. Так выглядит пример запроса (
SELECT) с дополнительно заданным максимальным числом строк:$rv = spi_exec_query('SELECT * FROM my_table', 5);Этот запрос возвращает не больше 5 строк из таблицы
my_table. Если вmy_tableесть столбецmy_column, получить его значение из строки$iрезультата можно следующим образом:$foo = $rv->{rows}[$i]->{my_column};Общее число строк, возвращённых запросом
SELECT, можно получить так:$nrows = $rv->{processed}Так можно выполнить команду другого типа:
$query = "INSERT INTO my_table VALUES (1, 'test')"; $rv = spi_exec_query($query);
Затем можно получить статус команды (например,
SPI_OK_INSERT) следующим образом:$res = $rv->{status};Чтобы получить число затронутых строк, выполните:
$nrows = $rv->{processed};Полный пример:
CREATE TABLE test ( i int, v varchar ); INSERT INTO test (i, v) VALUES (1, 'first line'); INSERT INTO test (i, v) VALUES (2, 'second line'); INSERT INTO test (i, v) VALUES (3, 'third line'); INSERT INTO test (i, v) VALUES (4, 'immortal'); CREATE OR REPLACE FUNCTION test_munge() RETURNS SETOF test AS $$ my $rv = spi_exec_query('select i, v from test;'); my $status = $rv->{status}; my $nrows = $rv->{processed}; foreach my $rn (0 .. $nrows - 1) { my $row = $rv->{rows}[$rn]; $row->{i} += 200 if defined($row->{i}); $row->{v} =~ tr/A-Za-z/a-zA-Z/ if (defined($row->{v})); return_next($row); } return undef; $$ LANGUAGE plperl; SELECT * FROM test_munge();-
spi_query(команда)
spi_fetchrow(cursor)
spi_cursor_close(cursor) Функции
spi_queryиspi_fetchrowприменяются в паре, когда набор строк может быть очень большим или когда нужно возвращать строки по мере их поступления. Функцияspi_fetchrowработает только сspi_query. Следующий пример показывает, как использовать их вместе:CREATE TYPE foo_type AS (the_num INTEGER, the_text TEXT); CREATE OR REPLACE FUNCTION lotsa_md5 (INTEGER) RETURNS SETOF foo_type AS $$ use Digest::MD5 qw(md5_hex); my $file = '/usr/share/dict/words'; my $t = localtime; elog(NOTICE, "opening file $file at $t" ); open my $fh, '<', $file # здесь мы обращаемся к файлу! or elog(ERROR, "cannot open $file for reading: $!"); my @words = <$fh>; close $fh; $t = localtime; elog(NOTICE, "closed file $file at $t"); chomp(@words); my $row; my $sth = spi_query("SELECT * FROM generate_series(1,$_[0]) AS b(a)"); while (defined ($row = spi_fetchrow($sth))) { return_next({ the_num => $row->{a}, the_text => md5_hex($words[rand @words]) }); } return; $$ LANGUAGE plperlu; SELECT * from lotsa_md5(500);Обычно вызов
spi_fetchrowнужно повторять, пока не будет получен результатundef, показывающий, что все строки уже прочитаны. Курсор, возвращаемый функциейspi_query, автоматически освобождается, когдаspi_fetchrowвозвращаетundef. Если вы не хотите читать все строки, освободите курсор, выполнивspi_cursor_close, чтобы не допустить утечки памяти.-
spi_prepare(команда,типы аргументов)
spi_query_prepared(план,аргументы)
spi_exec_prepared(план[,атрибуты],аргументы)
spi_freeplan(план) Функции
spi_prepare,spi_query_prepared,spi_exec_preparedиspi_freeplanреализуют ту же функциональность, но для подготовленных запросов. Функцияspi_prepareпринимает строку запроса с нумерованными местозаполнителями аргументов ($1, $2 и т. д.) и список строк с типами аргументов:$plan = spi_prepare('SELECT * FROM test WHERE id > $1 AND name = $2', 'INTEGER', 'TEXT');План запроса, подготовленный вызовом
spi_prepare, можно использовать вместо строки запроса либо вspi_exec_prepared, возвращающей тот же результат, что иspi_exec_query, либо вspi_query_prepared, возвращающей курсор так же, какspi_query, который затем можно передать вspi_fetchrow. В необязательном втором параметреspi_exec_preparedможно передать хеш с атрибутами; в настоящее время поддерживается только атрибутlimit, задающий максимальное число строк, которое может вернуть запрос. Без указанияlimitили при указании его равным нулю ограничение на количество строк снимается.Подготовленные запросы хороши тем, что позволяют использовать единожды подготовленный план для неоднократного выполнения запроса. Когда план оказывается не нужен, его можно освободить, вызвав
spi_freeplan:CREATE OR REPLACE FUNCTION init() RETURNS VOID AS $$ $_SHARED{my_plan} = spi_prepare('SELECT (now() + $1)::date AS now', 'INTERVAL'); $$ LANGUAGE plperl; CREATE OR REPLACE FUNCTION add_time( INTERVAL ) RETURNS TEXT AS $$ return spi_exec_prepared( $_SHARED{my_plan}, $_[0] )->{rows}->[0]->{now}; $$ LANGUAGE plperl; CREATE OR REPLACE FUNCTION done() RETURNS VOID AS $$ spi_freeplan( $_SHARED{my_plan}); undef $_SHARED{my_plan}; $$ LANGUAGE plperl; SELECT init(); SELECT add_time('1 day'), add_time('2 days'), add_time('3 days'); SELECT done(); add_time | add_time | add_time ------------+------------+------------ 2005-12-10 | 2005-12-11 | 2005-12-12Заметьте, что параметры для
spi_prepareобозначаются как $1, $2, $3 и т. д., так что по возможности не записывайте строки запросов в двойных кавычках, чтобы не спровоцировать трудноуловимые ошибки.Ещё один пример, иллюстрирующий использование необязательного параметра
spi_exec_prepared:CREATE TABLE hosts AS SELECT id, ('192.168.1.'||id)::inet AS address FROM generate_series(1,3) AS id; CREATE OR REPLACE FUNCTION init_hosts_query() RETURNS VOID AS $$ $_SHARED{plan} = spi_prepare('SELECT * FROM hosts WHERE address << $1', 'inet'); $$ LANGUAGE plperl; CREATE OR REPLACE FUNCTION query_hosts(inet) RETURNS SETOF hosts AS $$ return spi_exec_prepared( $_SHARED{plan}, {limit => 2}, $_[0] )->{rows}; $$ LANGUAGE plperl; CREATE OR REPLACE FUNCTION release_hosts_query() RETURNS VOID AS $$ spi_freeplan($_SHARED{plan}); undef $_SHARED{plan}; $$ LANGUAGE plperl; SELECT init_hosts_query(); SELECT query_hosts('192.168.1.0/30'); SELECT release_hosts_query(); query_hosts ----------------- (1,192.168.1.1) (2,192.168.1.2) (2 rows)-
spi_commit()
spi_rollback() Эти функции фиксируют или откатывают текущую транзакцию. Они могут вызываться только в процедурах или в анонимных блоках кода (в команде
DO), вызываемых из кода верхнего уровня. (Заметьте, что выполнить SQL-командыCOMMITилиROLLBACKчерезspi_exec_queryили подобную функцию нельзя. Соответствующие операции могут выполняться только данными функциями.) После завершения одной транзакции следующая начинается автоматически, отдельной функции для этого нет.Пример:
CREATE PROCEDURE transaction_test1() LANGUAGE plperl AS $$ foreach my $i (0..9) { spi_exec_query("INSERT INTO test1 (a) VALUES ($i)"); if ($i % 2 == 0) { spi_commit(); } else { spi_rollback(); } } $$; CALL transaction_test1();
46.3.2. Вспомогательные функции в PL/Perl
-
elog(уровень,сообщение) Выдаёт служебное сообщение или сообщение об ошибке. Возможные уровни сообщений:
DEBUG(ОТЛАДКА),LOG(СООБЩЕНИЕ),INFO(ИНФОРМАЦИЯ),NOTICE(ЗАМЕЧАНИЕ),WARNING(ПРЕДУПРЕЖДЕНИЕ) иERROR(ОШИБКА). С уровнемERRORвыдаётся ошибка; если она не перехватывается окружающим кодом Perl, она распространяется в вызывающий запрос, что приводит к прерыванию текущей транзакции или подтранзакции. По сути то же самое делает командаdieязыка Perl. При использовании других уровней происходит просто вывод сообщения с заданным уровнем важности. Будут ли сообщения определённого уровня передаваться клиенту и/или записываться в журнал, определяется конфигурационными параметрами log_min_messages и client_min_messages. За дополнительными сведениями обратитесь к Главе 19.-
quote_literal(строка) Оформляет переданную строку для использования в качестве текстовой строки в SQL-операторе. Включённые в неё апострофы и обратная косая черта при этом дублируются. Заметьте, что
quote_literalвозвращает undef, когда получает аргумент undef; если такие аргументы возможны, часто лучше использоватьquote_nullable.-
quote_nullable(строка) Оформляет переданную строку для использования в качестве текстовой строки в SQL-операторе; либо, если поступает аргумент undef, возвращает строку «NULL» (без кавычек). Символы апостроф и обратная косая черта дублируются должным образом.
-
quote_ident(строка) Оформляет переданную строку для использования в качестве идентификатора в SQL-операторе. При необходимости идентификатор заключается в кавычки (например, если он содержит символы, недопустимые в открытом виде, или буквы в разном регистре). Если переданная строка содержит кавычки, они дублируются.
-
decode_bytea(строка) Возвращает неформатированные двоичные данные, представленные содержимым заданной строки, которая должна быть закодирована как
bytea.-
encode_bytea(строка) Возвращает закодированные в виде
byteaдвоичные данные, содержащиеся в переданной строке.-
encode_array_literal(массив)
encode_array_literal(массив,разделитель) Возвращает содержимое указанного массива в виде строки в формате массива (см. Подраздел 8.15.2). Возвращает значение аргумента неизменённым, если это не ссылка не массив. Разделитель элементов в строке массива по умолчанию — «
,» (если разделитель не определён или undef).-
encode_typed_literal(значение,имя_типа) Преобразует переменную Perl в значение типа данных, указанного во втором аргументе, и возвращает строковое представление этого значения. Корректно обрабатывает вложенные массивы и значения составных типов.
-
encode_array_constructor(массив) Возвращает содержимое переданного массива в виде строки в формате конструктора массива (см. Подраздел 4.2.12). Отдельные значения заключаются в кавычки функцией
quote_nullable. Возвращает значение аргумента, заключённое в кавычки функциейquote_nullable, если аргумент — не ссылка на массив.-
looks_like_number(строка) Возвращает значение true, если содержимое переданной строки похоже на число, по правилам Perl, и false в обратном случае. Возвращает undef для аргумента undef. Ведущие и замыкающие пробелы игнорируются. Строки
InfиInfinityсчитаются представляющими число (бесконечность).-
is_array_ref(аргумент) Возвращает значение true, если переданный аргумент можно воспринять как ссылку на массив, то есть это ссылка на
ARRAYилиPostgreSQL::InServer::ARRAY. В противном случае возвращает false.
46.3. Built-in Functions
46.3.1. Database Access from PL/Perl
Access to the database itself from your Perl function can be done via the following functions:
-
spi_exec_query(query[,limit]) spi_exec_queryexecutes an SQL command and returns the entire row set as a reference to an array of hash references. Iflimitis specified and is greater than zero, thenspi_exec_queryretrieves at mostlimitrows, much as if the query included aLIMITclause. Omittinglimitor specifying it as zero results in no row limit.You should only use this command when you know that the result set will be relatively small. Here is an example of a query (
SELECTcommand) with the optional maximum number of rows:$rv = spi_exec_query('SELECT * FROM my_table', 5);This returns up to 5 rows from the table
my_table. Ifmy_tablehas a columnmy_column, you can get that value from row$iof the result like this:$foo = $rv->{rows}[$i]->{my_column};The total number of rows returned from a
SELECTquery can be accessed like this:$nrows = $rv->{processed}Here is an example using a different command type:
$query = "INSERT INTO my_table VALUES (1, 'test')"; $rv = spi_exec_query($query);
You can then access the command status (e.g.,
SPI_OK_INSERT) like this:$res = $rv->{status};To get the number of rows affected, do:
$nrows = $rv->{processed};Here is a complete example:
CREATE TABLE test ( i int, v varchar ); INSERT INTO test (i, v) VALUES (1, 'first line'); INSERT INTO test (i, v) VALUES (2, 'second line'); INSERT INTO test (i, v) VALUES (3, 'third line'); INSERT INTO test (i, v) VALUES (4, 'immortal'); CREATE OR REPLACE FUNCTION test_munge() RETURNS SETOF test AS $$ my $rv = spi_exec_query('select i, v from test;'); my $status = $rv->{status}; my $nrows = $rv->{processed}; foreach my $rn (0 .. $nrows - 1) { my $row = $rv->{rows}[$rn]; $row->{i} += 200 if defined($row->{i}); $row->{v} =~ tr/A-Za-z/a-zA-Z/ if (defined($row->{v})); return_next($row); } return undef; $$ LANGUAGE plperl; SELECT * FROM test_munge();-
spi_query(command)
spi_fetchrow(cursor)
spi_cursor_close(cursor) spi_queryandspi_fetchrowwork together as a pair for row sets which might be large, or for cases where you wish to return rows as they arrive.spi_fetchrowworks only withspi_query. The following example illustrates how you use them together:CREATE TYPE foo_type AS (the_num INTEGER, the_text TEXT); CREATE OR REPLACE FUNCTION lotsa_md5 (INTEGER) RETURNS SETOF foo_type AS $$ use Digest::MD5 qw(md5_hex); my $file = '/usr/share/dict/words'; my $t = localtime; elog(NOTICE, "opening file $file at $t" ); open my $fh, '<', $file # ooh, it's a file access! or elog(ERROR, "cannot open $file for reading: $!"); my @words = <$fh>; close $fh; $t = localtime; elog(NOTICE, "closed file $file at $t"); chomp(@words); my $row; my $sth = spi_query("SELECT * FROM generate_series(1,$_[0]) AS b(a)"); while (defined ($row = spi_fetchrow($sth))) { return_next({ the_num => $row->{a}, the_text => md5_hex($words[rand @words]) }); } return; $$ LANGUAGE plperlu; SELECT * from lotsa_md5(500);Normally,
spi_fetchrowshould be repeated until it returnsundef, indicating that there are no more rows to read. The cursor returned byspi_queryis automatically freed whenspi_fetchrowreturnsundef. If you do not wish to read all the rows, instead callspi_cursor_closeto free the cursor. Failure to do so will result in memory leaks.-
spi_prepare(command,argument types)
spi_query_prepared(plan,arguments)
spi_exec_prepared(plan[,attributes],arguments)
spi_freeplan(plan) spi_prepare,spi_query_prepared,spi_exec_prepared, andspi_freeplanimplement the same functionality but for prepared queries.spi_prepareaccepts a query string with numbered argument placeholders ($1, $2, etc) and a string list of argument types:$plan = spi_prepare('SELECT * FROM test WHERE id > $1 AND name = $2', 'INTEGER', 'TEXT');Once a query plan is prepared by a call to
spi_prepare, the plan can be used instead of the string query, either inspi_exec_prepared, where the result is the same as returned byspi_exec_query, or inspi_query_preparedwhich returns a cursor exactly asspi_querydoes, which can be later passed tospi_fetchrow. The optional second parameter tospi_exec_preparedis a hash reference of attributes; the only attribute currently supported islimit, which sets the maximum number of rows returned from the query. Omittinglimitor specifying it as zero results in no row limit.The advantage of prepared queries is that is it possible to use one prepared plan for more than one query execution. After the plan is not needed anymore, it can be freed with
spi_freeplan:CREATE OR REPLACE FUNCTION init() RETURNS VOID AS $$ $_SHARED{my_plan} = spi_prepare('SELECT (now() + $1)::date AS now', 'INTERVAL'); $$ LANGUAGE plperl; CREATE OR REPLACE FUNCTION add_time( INTERVAL ) RETURNS TEXT AS $$ return spi_exec_prepared( $_SHARED{my_plan}, $_[0] )->{rows}->[0]->{now}; $$ LANGUAGE plperl; CREATE OR REPLACE FUNCTION done() RETURNS VOID AS $$ spi_freeplan( $_SHARED{my_plan}); undef $_SHARED{my_plan}; $$ LANGUAGE plperl; SELECT init(); SELECT add_time('1 day'), add_time('2 days'), add_time('3 days'); SELECT done(); add_time | add_time | add_time ------------+------------+------------ 2005-12-10 | 2005-12-11 | 2005-12-12Note that the parameter subscript in
spi_prepareis defined via $1, $2, $3, etc, so avoid declaring query strings in double quotes that might easily lead to hard-to-catch bugs.Another example illustrates usage of an optional parameter in
spi_exec_prepared:CREATE TABLE hosts AS SELECT id, ('192.168.1.'||id)::inet AS address FROM generate_series(1,3) AS id; CREATE OR REPLACE FUNCTION init_hosts_query() RETURNS VOID AS $$ $_SHARED{plan} = spi_prepare('SELECT * FROM hosts WHERE address << $1', 'inet'); $$ LANGUAGE plperl; CREATE OR REPLACE FUNCTION query_hosts(inet) RETURNS SETOF hosts AS $$ return spi_exec_prepared( $_SHARED{plan}, {limit => 2}, $_[0] )->{rows}; $$ LANGUAGE plperl; CREATE OR REPLACE FUNCTION release_hosts_query() RETURNS VOID AS $$ spi_freeplan($_SHARED{plan}); undef $_SHARED{plan}; $$ LANGUAGE plperl; SELECT init_hosts_query(); SELECT query_hosts('192.168.1.0/30'); SELECT release_hosts_query(); query_hosts ----------------- (1,192.168.1.1) (2,192.168.1.2) (2 rows)-
spi_commit()
spi_rollback() Commit or roll back the current transaction. This can only be called in a procedure or anonymous code block (
DOcommand) called from the top level. (Note that it is not possible to run the SQL commandsCOMMITorROLLBACKviaspi_exec_queryor similar. It has to be done using these functions.) After a transaction is ended, a new transaction is automatically started, so there is no separate function for that.Here is an example:
CREATE PROCEDURE transaction_test1() LANGUAGE plperl AS $$ foreach my $i (0..9) { spi_exec_query("INSERT INTO test1 (a) VALUES ($i)"); if ($i % 2 == 0) { spi_commit(); } else { spi_rollback(); } } $$; CALL transaction_test1();
46.3.2. Utility Functions in PL/Perl
-
elog(level,msg) Emit a log or error message. Possible levels are
DEBUG,LOG,INFO,NOTICE,WARNING, andERROR.ERRORraises an error condition; if this is not trapped by the surrounding Perl code, the error propagates out to the calling query, causing the current transaction or subtransaction to be aborted. This is effectively the same as the Perldiecommand. The other levels only generate messages of different priority levels. Whether messages of a particular priority are reported to the client, written to the server log, or both is controlled by the log_min_messages and client_min_messages configuration variables. See Chapter 19 for more information.-
quote_literal(string) Return the given string suitably quoted to be used as a string literal in an SQL statement string. Embedded single-quotes and backslashes are properly doubled. Note that
quote_literalreturns undef on undef input; if the argument might be undef,quote_nullableis often more suitable.-
quote_nullable(string) Return the given string suitably quoted to be used as a string literal in an SQL statement string; or, if the argument is undef, return the unquoted string "NULL". Embedded single-quotes and backslashes are properly doubled.
-
quote_ident(string) Return the given string suitably quoted to be used as an identifier in an SQL statement string. Quotes are added only if necessary (i.e., if the string contains non-identifier characters or would be case-folded). Embedded quotes are properly doubled.
-
decode_bytea(string) Return the unescaped binary data represented by the contents of the given string, which should be
byteaencoded.-
encode_bytea(string) Return the
byteaencoded form of the binary data contents of the given string.-
encode_array_literal(array)
encode_array_literal(array,delimiter) Returns the contents of the referenced array as a string in array literal format (see Section 8.15.2). Returns the argument value unaltered if it's not a reference to an array. The delimiter used between elements of the array literal defaults to "
," if a delimiter is not specified or is undef.-
encode_typed_literal(value,typename) Converts a Perl variable to the value of the data type passed as a second argument and returns a string representation of this value. Correctly handles nested arrays and values of composite types.
-
encode_array_constructor(array) Returns the contents of the referenced array as a string in array constructor format (see Section 4.2.12). Individual values are quoted using
quote_nullable. Returns the argument value, quoted usingquote_nullable, if it's not a reference to an array.-
looks_like_number(string) Returns a true value if the content of the given string looks like a number, according to Perl, returns false otherwise. Returns undef if the argument is undef. Leading and trailing space is ignored.
InfandInfinityare regarded as numbers.-
is_array_ref(argument) Returns a true value if the given argument may be treated as an array reference, that is, if ref of the argument is
ARRAYorPostgreSQL::InServer::ARRAY. Returns false otherwise.