Обработка исключений в PostgreSQL - PullRequest
32 голосов
/ 02 февраля 2011

Я новичок в PostgreSQL. Может кто-нибудь, пожалуйста, исправьте этот запрос.

BEGIN TRANSACTION;

BEGIN;
    CREATE TABLE "Logs"."Events"
    (
        EventId BIGSERIAL NOT NULL PRIMARY KEY,
        PrimaryKeyId bigint NOT NULL,
        EventDateTime date NOT NULL DEFAULT(now()),
        Action varchar(12) NOT NULL,
        UserId integer NOT NULL REFERENCES "Office"."Users"(UserId),
        PrincipalUserId varchar(50) NOT NULL DEFAULT(user)
    );

    CREATE TABLE "Logs"."EventDetails"
    (
        EventDetailId BIGSERIAL NOT NULL PRIMARY KEY,
        EventId bigint NOT NULL REFERENCES "Logs"."Events"(EventId),
        Resource varchar(64) NOT NULL,
        OldVal varchar(4000) NOT NULL,
        NewVal varchar(4000) NOT NULL
    );


    COMMIT TRANSACTION;
    RAISE NOTICE 'Task completed sucessfully.'
EXCEPTION;
    ROLLBACK TRANSACTION;
    RAISE ERROR @ErrorMessage, @LineNumber, @ErrorState --how to catch errors and display them????
END;

Вопросы:

  1. Как напечатать сообщение типа «ПЕЧАТЬ» в T-SQL?
  2. Как вызвать ошибки с информацией об исключении?

Ответы [ 4 ]

64 голосов
/ 11 августа 2011

Чтобы поймать сообщение об ошибке и его код:

do $$       
begin

    create table yyy(a int);
    create table yyy(a int); -- this will cause an error

exception when others then 

    raise notice 'The transaction is in an uncommittable state. '
                 'Transaction was rolled back';

    raise notice '% %', SQLERRM, SQLSTATE;

end; $$ 
language 'plpgsql';

Вы еще не нашли номер строки

ОБНОВЛЕНИЕ 16 апреля 2019

В соответствии с предложением Диего Скараваджи, для Postgres 9.2 и выше используйте GET STACKED DIAGNOSTICS :

do language plpgsql $$
declare
    v_state   TEXT;
    v_msg     TEXT;
    v_detail  TEXT;
    v_hint    TEXT;
    v_context TEXT;
begin

    create table yyy(a int);
    create table yyy(a int); -- this will cause an error

exception when others then 

    get stacked diagnostics
        v_state   = returned_sqlstate,
        v_msg     = message_text,
        v_detail  = pg_exception_detail,
        v_hint    = pg_exception_hint,
        v_context = pg_exception_context;

    raise notice E'Got exception:
        state  : %
        message: %
        detail : %
        hint   : %
        context: %', v_state, v_msg, v_detail, v_hint, v_context;

    raise notice E'Got exception:
        SQLSTATE: % 
        SQLERRM: %', SQLSTATE, SQLERRM;     

    raise notice '%', message_text; -- invalid. message_text is contextual to GET STACKED DIAGNOSTICS only

end; $$;

Результат:

NOTICE:  Got exception:
        state  : 42P07
        message: relation "yyy" already exists
        detail : 
        hint   : 
        context: SQL statement "create table yyy(a int)"
PL/pgSQL function inline_code_block line 11 at SQL statement
NOTICE:  Got exception:
        SQLSTATE: 42P07 
        SQLERRM: relation "yyy" already exists

ERROR:  column "message_text" does not exist
LINE 1: SELECT message_text
               ^
QUERY:  SELECT message_text
CONTEXT:  PL/pgSQL function inline_code_block line 33 at RAISE
SQL state: 42703

Помимо GET STACKED DIAGNOSTICS соответствует стандарту SQL, его диагностические переменные (например, message_text) контекстуальны только для GSD. Поэтому, если в вашей таблице есть поле с именем message_text, нет никакой вероятности, что GSD может помешать значению вашего поля.

Хотя номер строки по-прежнему отсутствует.

15 голосов
/ 03 февраля 2011

Используйте оператор DO , новую опцию в версии 9.0:

DO LANGUAGE plpgsql
$$
BEGIN
CREATE TABLE "Logs"."Events"
    (
        EventId BIGSERIAL NOT NULL PRIMARY KEY,
        PrimaryKeyId bigint NOT NULL,
        EventDateTime date NOT NULL DEFAULT(now()),
        Action varchar(12) NOT NULL,
        UserId integer NOT NULL REFERENCES "Office"."Users"(UserId),
        PrincipalUserId varchar(50) NOT NULL DEFAULT(user)
    );

    CREATE TABLE "Logs"."EventDetails"
    (
        EventDetailId BIGSERIAL NOT NULL PRIMARY KEY,
        EventId bigint NOT NULL REFERENCES "Logs"."Events"(EventId),
        Resource varchar(64) NOT NULL,
        OldVal varchar(4000) NOT NULL,
        NewVal varchar(4000) NOT NULL
    );

    RAISE NOTICE 'Task completed sucessfully.';    
END;
$$;
3 голосов
/ 02 февраля 2011

Вы можете написать это как скрипт psql, например,

START TRANSACTION;
CREATE TABLE ...
CREATE TABLE ...
COMMIT;
\echo 'Task completed sucessfully.'

и запустить с

psql -f somefile.sql

Выявление ошибок с параметрами невозможно в PostgreSQL напрямую. При переносе такого кода некоторые люди кодируют необходимую информацию в строке ошибки и анализируют ее при необходимости.

Все это работает немного по-другому, поэтому будьте готовы переучивать / переосмысливать / переписывать много вещей.

0 голосов
/ 10 октября 2018

Просто хочу добавить два моих цента к этому старому сообщению:

По моему мнению, почти все механизмы реляционных баз данных включают выполнение транзакции фиксации автоматически после выполнения команды DDL, даже если у вас autocommit = falseТаким образом, вам не нужно начинать транзакцию, чтобы избежать потенциального создания усеченного объекта, потому что это совершенно не нужно.

...