bump version
[sqlcipher.git] / test / upsert4.test
blobc4bcc0329d727b966bb24308972a0db135664e76
1 # 2018-04-17
3 # The author disclaims copyright to this source code.  In place of
4 # a legal notice, here is a blessing:
6 #    May you do good and not evil.
7 #    May you find forgiveness for yourself and forgive others.
8 #    May you share freely, never taking more than you give.
10 #***********************************************************************
12 # Test cases for UPSERT
14 set testdir [file dirname $argv0]
15 source $testdir/tester.tcl
16 set testprefix upsert4
18 foreach {tn sql} {
19   1 { CREATE TABLE t1(a INTEGER PRIMARY KEY, b, c UNIQUE) }
20   2 { CREATE TABLE t1(a INT PRIMARY KEY, b, c UNIQUE) }
21   3 { CREATE TABLE t1(a INT PRIMARY KEY, b, c UNIQUE) WITHOUT ROWID}
22 } {
23   reset_db
24   execsql $sql
26   do_execsql_test 1.$tn.0 {
27     INSERT INTO t1 VALUES(1, NULL, 'one');
28     INSERT INTO t1 VALUES(2, NULL, 'two');
29     INSERT INTO t1 VALUES(3, NULL, 'three');
30   }
31   
32   do_execsql_test 1.$tn.1 {
33     INSERT INTO t1 VALUES(1, NULL, 'xyz') ON CONFLICT DO NOTHING;
34     SELECT * FROM t1;
35   } {
36     1 {} one 2 {} two 3 {} three
37   }
38   
39   do_execsql_test 1.$tn.2 {
40     INSERT INTO t1 VALUES(4, NULL, 'two') ON CONFLICT DO NOTHING;
41     SELECT * FROM t1;
42   } {
43     1 {} one 2 {} two 3 {} three
44   }
45   
46   do_execsql_test 1.$tn.3 {
47     INSERT INTO t1 VALUES(4, NULL, 'two') ON CONFLICT (c) DO UPDATE SET b = 1;
48     SELECT * FROM t1;
49   } {
50     1 {} one 2 1 two 3 {} three
51   }
52   
53   do_execsql_test 1.$tn.4 {
54     INSERT INTO t1 VALUES(2, NULL, 'zero') ON CONFLICT (a) DO UPDATE SET b=2;
55     SELECT * FROM t1;
56   } {1 {} one 2 2 two 3 {} three}
58   do_catchsql_test 1.$tn.5 {
59     INSERT INTO t1 VALUES(2, NULL, 'zero') ON CONFLICT (a) 
60       DO UPDATE SET c = 'one';
61   } {1 {UNIQUE constraint failed: t1.c}}
63   do_execsql_test 1.$tn.6 {
64     SELECT * FROM t1;
65   } {1 {} one 2 2 two 3 {} three}
67   do_execsql_test 1.$tn.7 {
68     INSERT INTO t1 VALUES(2, NULL, 'zero') ON CONFLICT (a) 
69       DO UPDATE SET (b, c) = (SELECT 'x', 'y');
70     SELECT * FROM t1;
71   } {1 {} one 2 x y 3 {} three}
73   do_execsql_test 1.$tn.8 {
74     INSERT INTO t1 VALUES(1, NULL, NULL) ON CONFLICT (a) 
75       DO UPDATE SET (c, a) = ('four', 4);
76     SELECT * FROM t1 ORDER BY 1;
77   } {2 x y 3 {} three 4 {} four}
80 #-------------------------------------------------------------------------
81 # Test target analysis.
83 set rtbl(0) {0 {}}
84 set rtbl(1) {/1 .*failed.*/}
85 set rtbl(2) {1 {ON CONFLICT clause does not match any PRIMARY KEY or UNIQUE constraint}}
87 foreach {tn sql} {
88   1 { 
89       CREATE TABLE xyz(a INTEGER PRIMARY KEY, b, c, d);
90       CREATE UNIQUE INDEX xyz1 ON xyz(d, c, b COLLATE nocase);
91   }
93   2 { 
94       CREATE TABLE xyz(a INT PRIMARY KEY, b, c, d);
95       CREATE UNIQUE INDEX xyz1 ON xyz(d, c, b COLLATE nocase);
96   }
98   3 { 
99       CREATE TABLE xyz(a INT PRIMARY KEY, b, c, d) WITHOUT ROWID;
100       CREATE UNIQUE INDEX xyz1 ON xyz(d, c, b COLLATE nocase);
101   }
102 } {
103   reset_db
104   execsql $sql
105   do_execsql_test 2.$tn.1 {
106     INSERT INTO xyz VALUES(10, 1, 1, 'one');
107   }
110   foreach {tn2 oc res} {
111     1 "ON CONFLICT (b COLLATE nocase, c, d) DO NOTHING"   0
112     2 "ON CONFLICT (b, c, d) DO NOTHING"                  0
113     3 "ON CONFLICT (b, c COLLATE nocase, d) DO NOTHING"   2
114     4 "ON CONFLICT (a) DO NOTHING"                        1
115     5 "ON CONFLICT DO NOTHING"                            0
116     6 "ON CONFLICT (b, c, d) WHERE a!=0 DO NOTHING"       0
117     7 "ON CONFLICT (d, c, c) WHERE a!=0 DO NOTHING"       2
118     8 "ON CONFLICT (b COLLATE nocase, c COLLATE nocase, d) DO NOTHING"   2
119     9 "ON CONFLICT (b, c, d) WHERE b==45 DO NOTHING"      0
120   } {
122     do_catchsql_test 2.$tn.2.$tn2 "
123       INSERT INTO xyz VALUES(11, 1, 1, 'one') $oc
124     " $rtbl($res)
125   }
127   do_execsql_test 2.$tn.3 {
128     SELECT * FROM xyz;
129   } {10 1 1 one}
132 foreach {tn sql} {
133   1 {
134     CREATE TABLE abc(a INTEGER PRIMARY KEY, x, y);
135     CREATE UNIQUE INDEX abc1 ON abc(('x' || x) COLLATE nocase);
136   }
137   2 {
138     CREATE TABLE abc(a INT PRIMARY KEY, x, y);
139     CREATE UNIQUE INDEX abc1 ON abc(('x' || x) COLLATE nocase);
140   }
141   3 { 
142     CREATE TABLE abc(a INT PRIMARY KEY, x, y) WITHOUT ROWID;
143     CREATE UNIQUE INDEX abc1 ON abc(('x' || x) COLLATE nocase);
144   }
145 } {
146   reset_db
147   execsql $sql
148   do_execsql_test 3.$tn.1 {
149     INSERT INTO abc VALUES(1, 'one', 'two');
150   }
152   foreach {tn2 oc res} {
153     1 "ON CONFLICT DO NOTHING"                             0
154     2 "ON CONFLICT ('x' || x) DO NOTHING"                  0
155     3 "ON CONFLICT (('x' || x) COLLATE nocase) DO NOTHING" 0
156     4 "ON CONFLICT (('x' || x) COLLATE binary) DO NOTHING" 2
157     5 "ON CONFLICT (x || 'x') DO NOTHING"                  2
158     6 "ON CONFLICT ((('x' || x))) DO NOTHING"              0
159   } {
160     do_catchsql_test 3.$tn.2.$tn2 "
161       INSERT INTO abc VALUES(2, 'one', NULL) $oc;
162     " $rtbl($res)
163   }
165   do_execsql_test 3.$tn.3 {
166     SELECT * FROM abc
167   } {1 one two}
170 foreach {tn sql} {
171   1 {
172     CREATE TABLE abc(a INTEGER PRIMARY KEY, x, y);
173     CREATE UNIQUE INDEX abc1 ON abc(x) WHERE y>0;
174     CREATE UNIQUE INDEX abc2 ON abc(y) WHERE x='xyz' COLLATE nocase;
175   }
176 } {
177   reset_db
178   execsql $sql
179   do_execsql_test 4.$tn.1 {
180     INSERT INTO abc VALUES(1, 'one', 1);
181     INSERT INTO abc VALUES(2, 'two', 2);
182     INSERT INTO abc VALUES(3, 'xyz', 3);
183     INSERT INTO abc VALUES(4, 'XYZ', 4);
184   }
186   foreach {tn2 oc res} {
187     1 "ON CONFLICT DO NOTHING"                                 0
188     2 "ON CONFLICT(x) WHERE y>0 DO NOTHING"                    0
189     3 "ON CONFLICT(x) DO NOTHING"                              2
190     4 "ON CONFLICT(x) WHERE y>=0 DO NOTHING"                   2
191     5 "ON CONFLICT(y) WHERE x='xyz' COLLATE nocase DO NOTHING" 1
192   } {
193     do_catchsql_test 4.$tn.2.$tn2 "
194       INSERT INTO abc VALUES(5, 'one', 10) $oc
195     " $rtbl($res)
196   }
198   do_execsql_test 4.$tn.3 {
199     SELECT * FROM abc
200   } {1 one 1 2 two 2 3 xyz 3 4 XYZ 4}
202   foreach {tn2 oc res} {
203     1 "ON CONFLICT DO NOTHING"                                 0
204     2 "ON CONFLICT(y) WHERE x='xyz' COLLATE nocase DO NOTHING" 0
205     3 "ON CONFLICT(y) WHERE x='xyz' COLLATE binary DO NOTHING" 2
206     4 "ON CONFLICT(x) WHERE y>0 DO NOTHING"                    1
207   } {
208     do_catchsql_test 4.$tn.2.$tn2 "
209       INSERT INTO abc VALUES(5, 'xYz', 3) $oc
210     " $rtbl($res)
211   }
214 do_catchsql_test 5.0 {
215   CREATE TABLE w1(a INT PRIMARY KEY, x, y);
216   CREATE UNIQUE INDEX w1expr ON w1(('x' || x));
217   INSERT INTO w1 VALUES(2, 'one', NULL)
218     ON CONFLICT (('x' || x) COLLATE nocase) DO NOTHING;
219 } {1 {ON CONFLICT clause does not match any PRIMARY KEY or UNIQUE constraint}}
221 #-------------------------------------------------------------------------
222 # Test that ON CONFLICT constraint processing occurs before any REPLACE
223 # constraint processing.
225 foreach {tn sql} {
226   1 {
227     CREATE TABLE t1(a INTEGER PRIMARY KEY, b UNIQUE, c);
228   }
229   2 {
230     CREATE TABLE t1(a INT PRIMARY KEY, b UNIQUE, c);
231   }
232   3 {
233     CREATE TABLE t1(a INT PRIMARY KEY, b UNIQUE, c) WITHOUT ROWID;
234   }
235 } {
236   reset_db
237   execsql $sql
238   do_execsql_test 6.1.$tn {
239     INSERT INTO t1 VALUES(1, 1, 'one');
240     INSERT INTO t1 VALUES(2, 2, 'two');
241     INSERT OR REPLACE INTO t1 VALUES(1, 2, 'two') ON CONFLICT(b) DO NOTHING;
242     PRAGMA integrity_check;
243   } {ok}
246 foreach {tn sql} {
247   1 {
248     CREATE TABLE t1(a INTEGER PRIMARY KEY, b UNIQUE, c UNIQUE);
249   }
250 } {
251   reset_db
252   execsql $sql
254   do_execsql_test 6.2.$tn.1 {
255     INSERT INTO t1 VALUES(1, 1, 1);
256     INSERT INTO t1 VALUES(2, 2, 2);
257   }
259   do_execsql_test 6.2.$tn.2 {
260     INSERT OR REPLACE INTO t1 VALUES(3, 1, 1) ON CONFLICT(b) DO NOTHING;
261     SELECT * FROM t1;
262     PRAGMA integrity_check;
263   } {1 1 1 2 2 2 ok}
265   do_execsql_test 6.2.$tn.3 {
266     INSERT OR REPLACE INTO t1 VALUES(3, 2, 2) ON CONFLICT(c) DO NOTHING;
267     SELECT * FROM t1;
268     PRAGMA integrity_check;
269   } {1 1 1 2 2 2 ok}
271   do_execsql_test 6.2.$tn.2 {
272     INSERT OR REPLACE INTO t1 VALUES(3, 1, 1) ON CONFLICT(b) 
273       DO UPDATE SET b=b||'x';
274     SELECT * FROM t1;
275     PRAGMA integrity_check;
276   } {1 1x 1 2 2 2 ok}
278   do_execsql_test 6.2.$tn.2 {
279     INSERT OR REPLACE INTO t1 VALUES(3, 2, 2) ON CONFLICT(c) 
280       DO UPDATE SET c=c||'x';
281     SELECT * FROM t1;
282     PRAGMA integrity_check;
283   } {1 1x 1 2 2 2x ok}
286 #-------------------------------------------------------------------------
287 # Test references to "excluded". And using an alias in an INSERT 
288 # statement.
290 foreach {tn sql} {
291   1 {
292     CREATE TABLE t1(w, x, y, z, PRIMARY KEY(x, y));
293     CREATE UNIQUE INDEX zz ON t1(z);
294   }
295   2 {
296     CREATE TABLE t1(w, x, y, z, PRIMARY KEY(x, y)) WITHOUT ROWID;
297     CREATE UNIQUE INDEX zz ON t1(z);
298   }
299 } {
300   reset_db
301   execsql $sql
302   do_execsql_test 7.$tn.0 {
303     INSERT INTO t1 VALUES('a', 1, 1, 1);
304     INSERT INTO t1 VALUES('b', 2, 2, 2);
305   }
307   do_execsql_test 7.$tn.1 {
308     INSERT INTO t1 VALUES('c', 3, 3, 1) ON CONFLICT(z) 
309       DO UPDATE SET w = excluded.w;
310     SELECT * FROM t1;
311   } {c 1 1 1 b 2 2 2}
313   do_execsql_test 7.$tn.2 {
314     INSERT INTO t1 VALUES('c', 2, 2, 3) ON CONFLICT(y, x) 
315       DO UPDATE SET w = w||w;
316     SELECT * FROM t1;
317   } {c 1 1 1 bb 2 2 2}
319   do_execsql_test 7.$tn.3 {
320     INSERT INTO t1 VALUES('c', 2, 2, 3) ON CONFLICT(y, x) 
321       DO UPDATE SET w = w||t1.w;
322     SELECT * FROM t1;
323   } {c 1 1 1 bbbb 2 2 2}
325   do_execsql_test 7.$tn.4 {
326     INSERT INTO t1 AS tbl VALUES('c', 2, 2, 3) ON CONFLICT(y, x) 
327       DO UPDATE SET w = w||tbl.w;
328     SELECT * FROM t1;
329   } {c 1 1 1 bbbbbbbb 2 2 2}
332 foreach {tn sql} {
333   1 {
334     CREATE TABLE excluded(w, x INTEGER, 'a b', z, PRIMARY KEY(x, 'a b'));
335     CREATE UNIQUE INDEX zz ON excluded(z);
336     CREATE INDEX zz2 ON excluded(z);
337   }
338   2 {
339     CREATE TABLE excluded(w, x, 'a b', z, PRIMARY KEY(x, 'a b')) WITHOUT ROWID;
340     CREATE UNIQUE INDEX zz ON excluded(z);
341     CREATE INDEX zz2 ON excluded(z);
342   }
343 } {
344   reset_db
345   execsql $sql
346   do_execsql_test 8.$tn.0 {
347     INSERT INTO excluded VALUES('a', 1, 1, 1);
348     INSERT INTO excluded VALUES('b', 2, 2, 2);
349   }
351   # Note: An error in Postgres: "table reference "excluded" is ambiguous".
352   #
353   do_execsql_test 8.$tn.1 {
354     INSERT INTO excluded VALUES('hello', 1, 1, NULL) ON CONFLICT(x, "a b")
355       DO UPDATE SET w=excluded.w;
356     SELECT * FROM excluded;
357   } {a 1 1 1 b 2 2 2}
359   do_execsql_test 8.$tn.2 {
360     INSERT INTO excluded AS x1 VALUES('hello', 1, 1, NULL) ON CONFLICT(x, [a b])
361       DO UPDATE SET w=excluded.w;
362     SELECT * FROM excluded;
363   } {hello 1 1 1 b 2 2 2}
365   do_execsql_test 8.$tn.3 {
366     INSERT INTO excluded AS x1 VALUES('hello', 1, 1, NULL) ON CONFLICT(x, [a b])
367       DO UPDATE SET w=w||w WHERE excluded.w!='hello';
368     SELECT * FROM excluded;
369   } {hello 1 1 1 b 2 2 2}
371   do_execsql_test 8.$tn.4 {
372     INSERT INTO excluded AS x1 VALUES('hello', 1, 1, NULL) ON CONFLICT(x, [a b])
373       DO UPDATE SET w=w||w WHERE excluded.x=1;
374     SELECT * FROM excluded;
375   } {hellohello 1 1 1 b 2 2 2}
377   do_catchsql_test 8.$tn.5 {
378     INSERT INTO excluded AS x1 VALUES('hello', 1, 1, NULL) 
379       ON CONFLICT(x, [a b]) WHERE y=1
380       DO UPDATE SET w=w||w WHERE excluded.x=1;
381   } {1 {no such column: y}}
384 #--------------------------------------------------------------------------
386 do_execsql_test 9.0 {
387   CREATE TABLE v(x INTEGER);
388   CREATE TABLE hist(x INTEGER PRIMARY KEY, cnt INTEGER);
389   CREATE TRIGGER vt AFTER INSERT ON v BEGIN
390     INSERT INTO hist VALUES(new.x, 1) ON CONFLICT(x) DO
391       UPDATE SET cnt=cnt+1;
392   END;
395 do_execsql_test 9.1 {
396   INSERT INTO v VALUES(1), (4), (1), (5), (5), (8), (9), (1);
397   SELECT * FROM hist;
398 } {
399   1 3
400   4 1
401   5 2
402   8 1
403   9 1
407 finish_test