F.84. utl_smtp — отправка электронных писем по протоколу SMTP #

utl_smtp — это расширение Postgres Pro, предназначенное для отправки электронных писем по протоколу SMTP из PL/pgSQL. Функциональность, предоставляемая этим модулем, во многом пересекается с функциональностью пакета UTL_SMTP в Oracle.

F.84.1. Установка #

Расширение utl_smtp включено в состав Postgres Pro Enterprise как стандартное расширение. Чтобы задействовать utl_smtp, создайте расширение с помощью следующего запроса:

CREATE EXTENSION utl_smtp;

F.84.2. Использование #

Чтобы отправить электронное письмо с помощью расширения utl_smtp, необходимо вызывать функции в определённом порядке:

  • Сначала откройте соединение, вызвав функцию open_connection.

  • Затем отправьте команду HELO/EHLO SMTP-серверу, используя функции helo/ehlo соответственно.

  • Отправьте адреса отправителя и получателя, используя функции mail и rcpt.

  • Вызовите функцию opendata, чтобы начать процесс отправки данных посредством отправки команды DATA SMTP-серверу.

  • После этого можно многократно вызывать функции write_data и write_raw_data для отправки фактических данных.

  • Процесс отправки данных завершается вызовом функции close_data.

  • После вызова open_data можно вызывать только следующие функции: write_data, write_raw_data или close_data. Вызов других функций приведёт к возникновению ошибки.

  • Кроме того, процесс отправки данных можно упростить, однократно вызвав функцию data.

  • После вызова close_data или data электронное письмо отправлено — вызовите функцию quit, чтобы завершить соединение.

F.84.3. Типы данных #

Расширение utl_smtp предоставляет следующие типы данных:

  • Тип reply представляет собой строку ответа SMTP. Каждая строка ответа SMTP состоит из кода ответа, за которым следует текстовое сообщение. Для большинства команд SMTP ожидается одна строка ответа, для некоторых — несколько строк.

    Таблица F.147. Параметры reply

    ПараметрОписание
    code3-значный код ответа
    intТекстовое сообщение ответа

  • Тип connection представляет собой соединение по протоколу SMTP.

    Таблица F.148. Параметры connection

    ПараметрОписание
    hostИмя или IP-адрес удалённого узла при установке соединения.
    portНомер порта подключённого удалённого SMTP-сервера.
    tx_timeoutВремя ожидания расширением utl_smtp операций чтения или записи в этом соединении, в секундах. Таймаут соединения с сервером всегда составляет 60 секунд и не может быть настроен.
    private_socketЭтот параметр используется внутри Postgres Pro и не должен изменяться вручную.

F.84.4. Функции utl_smtp #

Расширение utl_smtp предоставляет функции для отправки электронных писем по протоколу SMTP. Обратите внимание, что функции, которые должны возвращать несколько строк ответа, возвращают массив типа reply.

auth(c connection, username text, password text, schemes text default 'PLAIN') returns reply #

Отправляет SMTP-серверу команду AUTH для аутентификации. В настоящее время поддерживается только схема аутентификации PLAIN.

close_all_connections() returns void #

Закрывает все SMTP-соединения и освобождает все связанные с ними ресурсы.

close_connection(c connection) returns void #

Закрывает указанное SMTP-соединение. Эту функцию можно вызвать при отправке данных на сервер перед вызовом close_data. В этом случае следующий вызов функции с этим соединением выдаст исключение.

close_data(c connection) returns reply #

Завершает сообщение электронной почты отправкой последовательности <CR><LF>.<CR><LF> (одна точка в начале строки).

command(c connection, cmd text, arg text default null) returns reply #

Отправляет произвольную команду SMTP и может возвращать несколько строк ответа.

data(c connection, body text) returns reply #

Указывает тело электронного письма. По сути, это последовательность вызовов следующих функций: open_data, write_data и close_data.

ehlo(c connection, domain text) returns reply #

Выполняет начальное согласование с SMTP-сервером с помощью команды EHLO. Сервер возвращает часть своей конфигурации.

helo(c connection, domain text) returns reply #

Выполняет начальное согласование с SMTP-сервером с помощью команды HELO.

help(c connection, command text default null) returns reply #

Отправляет команду HELP. Эта команда может быть реализована не на всех SMTP-серверах.

last_reply(c connection) returns reply #

Возвращает последний ответ SMTP-сервера.

mail(c connection, sender text, parameters text default null) returns reply #

Инициирует транзакцию отправки электронного письма с сервером, отправляя команду MAIL с адресом отправителя.

noop(c connection) returns reply #

Выдаёт команду NOOP. Эта функция в основном используется для проверки соединения.

open_connection(host text, port int default 25, tx_timeout int default null, secure_connection_before_smtp bool default false, verify_peer bool default true) returns connection #

Открывает соединение с SMTP-сервером. Параметр secure_connection_before_smtp указывает, устанавливается ли подключение TLS до соединения SMTP (по сути, если этот параметр имеет значение true, соединение будет использовать SMTPS вместо SMTP). Параметр verify_peer указывает, проверяются ли сертификаты при установке подключения TLS.

open_data(c connection) returns reply #

Отправляет команду DATA, после которой можно использовать функции write_data и write_raw_data, чтобы написать часть электронного письма.

quit(c connection) returns reply #

Завершает сеанс SMTP и отключается от сервера.

rcpt(c connection, recipient text, parameters text default null) returns reply #

Указывает получателя электронного письма. Транзакция отправки сообщения должна быть запущена предыдущим вызовом MAIL, а соединение с почтовым сервером должно быть открыто и инициализировано предыдущими вызовами open_connection и helo или ehlo соответственно.

rset(c connection) returns reply #

Завершает текущую транзакцию отправки письма. Клиент может вызвать rset в любое время после открытия соединения с SMTP-сервером с помощью open_connection до вызова data или open_data.

set_reply_error_check(c connection, enable bool) returns void #

Определяет поведение функции. Если для параметра enable установлено значение true (по умолчанию), любая ошибка на SMTP-сервере выдаёт исключение. Если установлено значение false, для определения ошибки пользователю следует самостоятельно анализировать результаты, возвращаемые функциями.

starttls(c connection, verify_peer bool default true) returns reply #

Отправляет команду STARTTLS для защиты SMTP-соединения с использованием TLS.

vrfy(c connection, recipient text) returns reply #

Отправляет команду VRFY для проверки корректности адреса получателя электронного письма. Эта команда может быть реализована не на всех SMTP-серверах и может возвращать некорректную информацию, поэтому не рекомендуется её использовать.

write_data(c connection, data text) returns void #

Отправляет часть текста сообщения, включая заголовки, SMTP-серверу. При повторном вызове write_data добавляет данные к сообщению.

write_raw_data(c connection, data text) returns void #

Отправляет часть текста сообщения, включая заголовки, SMTP-серверу. При повторном вызове write_data добавляет данные к сообщению. Эта функция аналогична функции write_data.

F.84.5. Примеры #

В следующем примере показана отправка электронного письма без вложения с помощью utl_smtp.

DO $$
DECLARE
    conn utl_smtp.connection;
BEGIN
    conn := utl_smtp.open_connection('smtp.mail.ru', 25, 10);
    perform utl_smtp.ehlo(conn, 'localhost');
    perform utl_smtp.starttls(conn);
    perform utl_smtp.ehlo(conn, 'localhost');
    perform utl_smtp.auth(conn, 'test_email@example.com', 'super-secret-password');
    perform utl_smtp.mail(conn, 'sender@example.com');
    perform utl_smtp.rcpt(conn, 'recipient@example.com');
    perform utl_smtp.open_data(conn);
    perform utl_smtp.write_data(conn, E'Content-Type: multipart/mixed; boundary=------------------------6f48b7d5ded0c5fc\n');
    perform utl_smtp.write_data(conn, E'Mime-Version: 1.0\n');
    perform utl_smtp.write_data(conn, E'From: Sender <sender@example.com>\n');
    perform utl_smtp.write_data(conn, E'To: Recipient <recipient@example.com>\n');
    perform utl_smtp.write_data(conn, E'Subject: mail from utl_smtp\n');
    perform utl_smtp.write_data(conn, E'--------------------------6f48b7d5ded0c5fc\n');
    perform utl_smtp.write_data(conn, E'Content-Type: text/plain; charset=\"UTF-8\"\n');
    perform utl_smtp.write_data(conn, E'Content-Transfer-Encoding: 8bit\n\n');
    perform utl_smtp.write_data(conn, E'This is body from inside Postgres Pro\n');
    perform utl_smtp.write_data(conn, E'Sent using utl_smtp\n');
    perform utl_smtp.write_data(conn, E'\n--------------------------6f48b7d5ded0c5fc--\n');
    perform utl_smtp.close_data(conn);
    perform utl_smtp.quit(conn);
END$$;

В следующем примере показана обработка исключений с помощью utl_smtp.

DO $$
DECLARE
    ...
    r utl_smtp.reply;
BEGIN
    ...
    some utl_smtp funcion calls
    ...
exception
    when others then
        r = utl_smtp.last_reply(conn);
        if r.code >= 500 then
            raise notice 'caught permanent error';
        elsif r.code >= 400 then
            raise notice 'caught transient error';
        else
            raise notice 'some other error';
        end if;
END$$;