2017-05-30 202 views
2

所以我有一个表类似的结构:试图UPSERT一个表部分索引

CREATE TABLE x (
    id SERIAL, 
    a character varying(1024) NOT NULL, 
    b character varying(2048), 
    c character varying(1024) 
); 

CREATE UNIQUE INDEX uniq_x_a ON x USING btree (a) WHERE (b IS NULL); 
CREATE UNIQUE INDEX uniq_x_a_b ON x USING btree (a, b) WHERE (b IS NOT NULL); 

现在我升级这种情况下,以9.5和要使用的ON CONFLICT DO UPDATE

所以执行这个

INSERT INTO x (a, b, c) VALUES ('hello', 'there', 'c1'); 
INSERT INTO x (a, b, c) VALUES ('hello', 'there', 'c2'); 

现在给我

ERROR: duplicate key value violates unique constraint "uniq_x_a_b" 
DETAIL: Key (a, b)=(hello, there) already exists. 

后来,当我做

INSERT INTO x (a, b, c) VALUES ('hello', 'there', 'c4') 
ON CONFLICT ON CONSTRAINT uniq_x_a_b DO UPDATE SET c = excluded.c; 

我得到

ERROR: constraint "uniq_x_a_b" for table "x" does not exist 

我不明白,我违反了一个不存在的约束?任何人都有小费?

编辑

每我加

ALTER TABLE x ADD CONSTRAINT uniq_x_a_b_constraint UNIQUE (a, b); 

所以这个现在适用于以下几种情况下建议:

INSERT INTO x (a, b, c) VALUES ('hello', 'there', 'c3') 
ON CONFLICT ON CONSTRAINT uniq_x_a_b_constraint DO UPDATE SET c = excluded.c; 

但失败了

INSERT INTO x (a, b, c) VALUES ('hello', NULL, 'c3') 
ON CONFLICT ON CONSTRAINT uniq_x_a_b_constraint DO UPDATE SET c = excluded.c; 

ERROR: duplicate key value violates unique constraint "uniq_x_a" 
DETAIL: Key (a)=(hello) already exists. 

我不能单独a列添加一个唯一约束,因为它是被认为是唯一的ab组合。似乎也不可能像创建索引一样在行的子集上构建约束。

回答

2

这里有一些东西在玩。

  1. 创建唯一索引不会自动创建一个唯一约束(尽管术语constraint错误消息在使用时你想插入重复记录)。另一方面,添加唯一约束将自动在列或列组上创建唯一的B树索引(请参阅此doc)。

  2. 不同于常规指标的情况下,限制不能使用的部分索引补充说:

    ALTER TABLE x ADD CONSTRAINT uniq_x_a_constraint UNIQUE USING INDEX uniq_x_a; 
    ERROR: "uniq_x_a" is a partial index ... 
    DETAIL: Cannot create a primary key or unique constraint using such an index. 
    

在本质上,你需要在(a, b)(无关创建一个单独的唯一约束部分索引)以便以预期的方式生成ON CONFLICT函数。

[更新]

即使a和b的组合是唯一的,ON CONFLICT将不会看到任何有效的约束强制唯一性:

INSERT INTO x (a, b, c) VALUES ('hello', 'there', 'c4') 
ON CONFLICT (a, b) DO UPDATE SET c = excluded.c; 
ERROR: there is no unique or exclusion constraint matching the ON CONFLICT specification 

现在,如果你添加约束,既ON CONFLICT (a, b)ON CONFLICT ON CONSTRAINT将工作:

ALTER TABLE x ADD CONSTRAINT uniq_x_a_b_constraint UNIQUE (a, b); 

INSERT INTO x (a, b, c) VALUES ('hello', 'there', 'c4') 
ON CONFLICT (a, b) DO UPDATE SET c = excluded.c; 

INSERT INTO x (a, b, c) VALUES ('hello', 'there', 'c5') 
ON CONFLICT ON CONSTRAINT uniq_x_a_b_constraint DO UPDATE SET c = excluded.c; 
+0

a和b的组合应该是唯一的,只有b可以包含NULL值。 – Jan

+0

@Jan,请参阅我的扩展答案。 –

+0

是的,但是这在'INSERT INTO x(a,b,c)VALUES('hello',NULL,'c3')上失败ON CONFLICT ON CONSTRAINT uniq_x_a_b_constraint DO UPDATE SET c = excluded.c;' – Jan