在触发程序中什么都不做

时间:2016-11-21 14:55:34

标签: postgresql triggers duplicates plpgsql upsert

尝试执行触发器时遇到了麻烦。假设我们有2个表,我想将数据从表A 复制到表B ,但每个表都有唯一约束

create table test1 (
 test_name varchar);

create unique index test1_uc on test1 USING btree (test_name);

create table test2 (
test_name2 varchar);

 create unique index test2_uc on test2 USING btree (test_name2);

CREATE OR REPLACE FUNCTION trig_test()
  RETURNS trigger AS
$$
BEGIN
  IF pg_trigger_depth() <> 1 THEN
    RETURN NEW;
END IF;
INSERT INTO test2(test_name2)
   VALUES(NEW.test_name2)
ON CONFLICT (test_name2) DO NOTHING;

RETURN NEW;
END;
$$ LANGUAGE plpgsql;

CREATE TRIGGER trigger_test
AFTER INSERT
ON test2
FOR EACH ROW
EXECUTE PROCEDURE trig_test();

insert into test2 values ('test');
insert into test2 values ('test'); //should do nothing ?

但是我收到了这个错误:

ERROR:  duplicate key value violates unique constraint "test2_uc"
DETAIL:  Key (test_name2)=(test) already exists.

触发器有什么问题?

1 个答案:

答案 0 :(得分:0)

你的例子坏了。触发器中的INSERT中的源和目标是相同的,每次(插入NULL时除外)都会引发唯一的违规 - 被ON CONFLICT (test_name2) DO NOTHING抑制,所以触发器中什么都没发生。

您还忘记了原始INSERT中的唯一约束。见下文。

INSERT INTO test2(test_name2)
   VALUES(NEW.test_name2)

...

CREATE TRIGGER trigger_test
AFTER INSERT
ON test2

从较少混乱的设置开始:

CREATE TABLE test1 (col1 text UNIQUE);
CREATE TABLE test2 (col2 text UNIQUE);

pg_trigger_depth()移动到触发器本身更有效。所以这可行,将插入test1的行复制到test2(而不是其他方式),仅用于第一个级别的触发深度:

CREATE OR REPLACE FUNCTION trig_test()
  RETURNS trigger AS
$func$
BEGIN
   INSERT INTO test2(col2)             -- !!
   VALUES (NEW.col1)                   -- !!
   ON     CONFLICT (col2) DO NOTHING;  -- !!

   RETURN NULL;
END
$func$ LANGUAGE plpgsql;

我将其保留为AFTER触发器。也可以是BEFORE触发器,但您需要 RETURN NEW;

CREATE TRIGGER trigger_test
AFTER INSERT ON test1                  -- !!
FOR EACH ROW 
WHEN (pg_trigger_depth() < 1)          -- !!
EXECUTE PROCEDURE trig_test();

为什么(pg_trigger_depth() < 1)

注意 您以test2这种方式捕获了唯一的违规行为(没有任何反应),但test1中的唯一违规行为仍会引发异常除非你有ON CONFLICT ... DO NOTHING。你的考试是一厢情愿的想法:

insert into test2 values ('test'); //should do nothing ?

必须:

INSERT INTO test1 values ('test') ON CONFLICT (col1) DO NOTHING;

替代方案:使用CTE链接两个INSERT

如果您可以控制INSERT上的test1命令,则可以执行此操作而不是触发器:

WITH ins1 AS (
   INSERT INTO test1(col1)
   VALUES ('foo')                  -- your value goes here
   ON CONFLICT (col1) DO NOTHING
   RETURNING *
   )
INSERT INTO test2(col2)
SELECT col1 FROM ins1
ON CONFLICT (col2) DO NOTHING;

相关: