Идемпотентные сценарии PostgreSQL DDL - PullRequest
4 голосов
/ 09 апреля 2009

Я ищу способ сценария изменения схемы postgreSQL идемпотентным способом.

В MSSQL я мог бы сделать что-то вроде этого:

if(not exists(select * from information_schema.columns where table_name = 'x' and column_name = 'y'))
begin
    alter table x add y int
end
go

PostgreSQL, похоже, не допускает ad-hoc pl / pgsql таким же образом, как MSSQL делает с T-SQL, поэтому я не могу использовать управляющие структуры в сценарии SQL и запускать его с помощью psql -f x.sql.

Я знаю, что PostgreSQL выдаст ошибку, если объект уже существует, но я не хочу игнорировать ошибки.

Я мог бы использовать некоторую технику управления версиями схемы, такую ​​как dbdeploy, но мне действительно нравится простота запуска набора файлов через psql без каких-либо нежелательных побочных эффектов.

Возможно ли это?

Спасибо, Mark

Ответы [ 3 ]

9 голосов
/ 28 июня 2017

Отказ от ответственности: я знаю, это очень старый вопрос, и уже есть принятый ответ. Но я бы хотел зарегистрировать здесь абсолютно идемпотентный сценарий без внешних ссылок. <Ч /> Простой скрипт, демонстрирующий встроенные возможности PostgreSQL 9.5+. Вы можете запускать этот скрипт столько раз, сколько пожелаете. Вот и мы:

--Table
CREATE TABLE IF NOT EXISTS person (
  id integer NOT NULL,
  person_name character varying(40) NOT NULL,
  updated_date date,
  CONSTRAINT person_pkey PRIMARY KEY (id)
);

--Index
CREATE INDEX IF NOT EXISTS idx_person_name ON person (person_name);

--Sequence
CREATE SEQUENCE IF NOT EXISTS seq_person_inc;

--Function
CREATE OR REPLACE FUNCTION simple_sum(a_integer int, b_integer int) RETURNS INT
    AS $$ SELECT a_integer+b_integer $$
LANGUAGE SQL;

--View
CREATE OR REPLACE VIEW vw_select_1 AS
    SELECT 1;

--Role
DO $$
BEGIN
    CREATE ROLE rick_deckard;    
EXCEPTION   
    WHEN duplicate_object THEN
        RAISE NOTICE 'Role already exists. Ignoring...';    
END$$;

--Simple insert
INSERT INTO person (id, person_name) VALUES (1, 'HAL-9000'); 

--Upsert (insert + update)
INSERT INTO person (id, person_name) VALUES (1, 'Betrayer') ON CONFLICT ON CONSTRAINT person_pkey DO UPDATE SET person_name = EXCLUDED.person_name;

--Upsert (ignoring duplicate error)
INSERT INTO person (id, person_name) VALUES (1, 'HAL-9000') ON CONFLICT ON CONSTRAINT person_pkey DO NOTHING;

--Upsert (ignoring any error)
INSERT INTO person (id, person_name) VALUES (1, 'HAL-9000') ON CONFLICT DO NOTHING;

--Field
DO $$
BEGIN
    ALTER TABLE person ADD COLUMN id_another_person INTEGER;
EXCEPTION   
    WHEN duplicate_column THEN
        RAISE NOTICE 'Field already exists. Ignoring...';
END$$;

--Constraint
DO $$
BEGIN
    ALTER TABLE person ADD CONSTRAINT person_id_another_person_fkey FOREIGN KEY (id_another_person) REFERENCES person (id);
EXCEPTION   
    WHEN duplicate_object THEN
        RAISE NOTICE 'Constraint already exists. Ignoring...';
END$$;

--Trigger
CREATE OR REPLACE FUNCTION person_trigger_function() RETURNS trigger AS $BODY$
BEGIN
   --Something complex here =)
   RETURN NEW; 
END;
$BODY$
LANGUAGE plpgsql;

DO $$
BEGIN
    CREATE TRIGGER person_trigger BEFORE INSERT OR UPDATE ON person FOR EACH ROW EXECUTE PROCEDURE person_trigger_function();
EXCEPTION   
    WHEN duplicate_object THEN
        RAISE NOTICE 'Trigger already exists. Ignoring...';
END$$;

--Drop
DROP TRIGGER IF EXISTS person_trigger ON person;
DROP INDEX IF EXISTS idx_person_name;
ALTER TABLE person DROP COLUMN IF EXISTS person_name;
ALTER TABLE person DROP CONSTRAINT IF EXISTS person_id_another_person_fkey;
DROP ROLE IF EXISTS rick_deckard;
DROP VIEW IF EXISTS vw_select_1;
DROP FUNCTION IF EXISTS simple_sum(integer, integer);
DROP FUNCTION IF EXISTS person_trigger_function();
DROP TABLE IF EXISTS person;
DROP SEQUENCE IF EXISTS seq_person_inc;
4 голосов
/ 09 апреля 2009

этот пост может оказаться полезным: http://www.depesz.com/index.php/2008/06/18/conditional-ddl/

2 голосов
/ 09 апреля 2009

Вы можете использовать plpgsql :

create language plpgsql;
create function f() ... as $$
<plpgsql code>
$$language plpgsql;
select f();
...