blob: 9a2444a87e82829b04f19360794c113edc2d0595 [file] [log] [blame]
drhd2694612013-11-04 22:04:17 +00001# 2013-11-04
2#
3# The author disclaims copyright to this source code. In place of
4# a legal notice, here is a blessing:
5#
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.
9#
10#***********************************************************************
11#
12# Test cases for partial indices in WITHOUT ROWID tables
13#
14
15
16set testdir [file dirname $argv0]
17source $testdir/tester.tcl
18
19ifcapable !vtab {
20 finish_test
21 return
22}
23
drh7be0fd92015-03-05 15:34:15 +000024# Capture the output of a pragma in a TEMP table.
25#
26proc capture_pragma {db tabname sql} {
27 $db eval "DROP TABLE IF EXISTS temp.$tabname"
28 set once 1
29 $db eval $sql x {
30 if {$once} {
31 set once 0
32 set ins "INSERT INTO $tabname VALUES"
33 set crtab "CREATE TEMP TABLE $tabname "
34 set sep "("
35 foreach col $x(*) {
36 append ins ${sep}\$x($col)
37 append crtab ${sep}\"$col\"
38 set sep ,
39 }
40 append ins )
41 append crtab )
42 $db eval $crtab
43 }
44 $db eval $ins
45 }
46}
47
48
drhd2694612013-11-04 22:04:17 +000049load_static_extension db wholenumber;
50do_test index7-1.1 {
51 # Able to parse and manage partial indices
52 execsql {
53 CREATE TABLE t1(a,b,c PRIMARY KEY) WITHOUT rowid;
54 CREATE INDEX t1a ON t1(a) WHERE a IS NOT NULL;
55 CREATE INDEX t1b ON t1(b) WHERE b>10;
56 CREATE VIRTUAL TABLE nums USING wholenumber;
57 INSERT INTO t1(a,b,c)
58 SELECT CASE WHEN value%3!=0 THEN value END, value, value
59 FROM nums WHERE value<=20;
60 SELECT count(a), count(b) FROM t1;
61 PRAGMA integrity_check;
62 }
63} {14 20 ok}
64
drh7be0fd92015-03-05 15:34:15 +000065# (The "partial" column of the PRAGMA index_list output is...)
66# EVIDENCE-OF: R-34457-09668 "1" if the index is a partial index and "0"
67# if not.
68#
69do_test index7-1.1a {
70 capture_pragma db out {PRAGMA index_list(t1)}
71 db eval {SELECT "name", "partial", '|' FROM out ORDER BY "name"}
72} {sqlite_autoindex_t1_1 0 | t1a 1 | t1b 1 |}
73
drhd2694612013-11-04 22:04:17 +000074# Make sure the count(*) optimization works correctly with
75# partial indices. Ticket [a5c8ed66cae16243be6] 2013-10-03.
76#
77do_execsql_test index7-1.1.1 {
78 SELECT count(*) FROM t1;
79} {20}
80
81# Error conditions during parsing...
82#
83do_test index7-1.2 {
84 catchsql {
85 CREATE INDEX bad1 ON t1(a,b) WHERE x IS NOT NULL;
86 }
87} {1 {no such column: x}}
88do_test index7-1.3 {
89 catchsql {
90 CREATE INDEX bad1 ON t1(a,b) WHERE EXISTS(SELECT * FROM t1);
91 }
92} {1 {subqueries prohibited in partial index WHERE clauses}}
93do_test index7-1.4 {
94 catchsql {
95 CREATE INDEX bad1 ON t1(a,b) WHERE a!=?1;
96 }
97} {1 {parameters prohibited in partial index WHERE clauses}}
98do_test index7-1.5 {
99 catchsql {
100 CREATE INDEX bad1 ON t1(a,b) WHERE a!=random();
101 }
102} {1 {functions prohibited in partial index WHERE clauses}}
103do_test index7-1.6 {
104 catchsql {
105 CREATE INDEX bad1 ON t1(a,b) WHERE a NOT LIKE 'abc%';
106 }
107} {1 {functions prohibited in partial index WHERE clauses}}
108
109do_test index7-1.10 {
110 execsql {
111 ANALYZE;
112 SELECT idx, stat FROM sqlite_stat1 ORDER BY idx;
113 PRAGMA integrity_check;
114 }
115} {t1 {20 1} t1a {14 1} t1b {10 1} ok}
116
117# STAT1 shows the partial indices have a reduced number of
118# rows.
119#
120do_test index7-1.11 {
121 execsql {
122 UPDATE t1 SET a=b;
123 ANALYZE;
124 SELECT idx, stat FROM sqlite_stat1 ORDER BY idx;
125 PRAGMA integrity_check;
126 }
127} {t1 {20 1} t1a {20 1} t1b {10 1} ok}
128
129do_test index7-1.11b {
130 execsql {
131 UPDATE t1 SET a=NULL WHERE b%3!=0;
132 UPDATE t1 SET b=b+100;
133 ANALYZE;
134 SELECT idx, stat FROM sqlite_stat1 ORDER BY idx;
135 PRAGMA integrity_check;
136 }
137} {t1 {20 1} t1a {6 1} t1b {20 1} ok}
138
139do_test index7-1.12 {
140 execsql {
141 UPDATE t1 SET a=CASE WHEN b%3!=0 THEN b END;
142 UPDATE t1 SET b=b-100;
143 ANALYZE;
144 SELECT idx, stat FROM sqlite_stat1 ORDER BY idx;
145 PRAGMA integrity_check;
146 }
147} {t1 {20 1} t1a {13 1} t1b {10 1} ok}
148
149do_test index7-1.13 {
150 execsql {
151 DELETE FROM t1 WHERE b BETWEEN 8 AND 12;
152 ANALYZE;
153 SELECT idx, stat FROM sqlite_stat1 ORDER BY idx;
154 PRAGMA integrity_check;
155 }
156} {t1 {15 1} t1a {10 1} t1b {8 1} ok}
157
158do_test index7-1.14 {
159 execsql {
160 REINDEX;
161 ANALYZE;
162 SELECT idx, stat FROM sqlite_stat1 ORDER BY idx;
163 PRAGMA integrity_check;
164 }
165} {t1 {15 1} t1a {10 1} t1b {8 1} ok}
166
167do_test index7-1.15 {
168 execsql {
169 CREATE INDEX t1c ON t1(c);
170 ANALYZE;
171 SELECT idx, stat FROM sqlite_stat1 ORDER BY idx;
172 PRAGMA integrity_check;
173 }
174} {t1 {15 1} t1a {10 1} t1b {8 1} t1c {15 1} ok}
175
176# Queries use partial indices as appropriate times.
177#
178do_test index7-2.1 {
179 execsql {
180 CREATE TABLE t2(a,b PRIMARY KEY) without rowid;
181 INSERT INTO t2(a,b) SELECT value, value FROM nums WHERE value<1000;
182 UPDATE t2 SET a=NULL WHERE b%5==0;
183 CREATE INDEX t2a1 ON t2(a) WHERE a IS NOT NULL;
184 SELECT count(*) FROM t2 WHERE a IS NOT NULL;
185 }
186} {800}
187do_test index7-2.2 {
188 execsql {
189 EXPLAIN QUERY PLAN
190 SELECT * FROM t2 WHERE a=5;
191 }
192} {/.* TABLE t2 USING COVERING INDEX t2a1 .*/}
193ifcapable stat4||stat3 {
194 do_test index7-2.3stat4 {
195 execsql {
196 EXPLAIN QUERY PLAN
197 SELECT * FROM t2 WHERE a IS NOT NULL;
198 }
199 } {/.* TABLE t2 USING COVERING INDEX t2a1 .*/}
200} else {
201 do_test index7-2.3stat4 {
202 execsql {
203 EXPLAIN QUERY PLAN
204 SELECT * FROM t2 WHERE a IS NOT NULL AND a>0;
205 }
206 } {/.* TABLE t2 USING COVERING INDEX t2a1 .*/}
207}
208do_test index7-2.4 {
209 execsql {
210 EXPLAIN QUERY PLAN
211 SELECT * FROM t2 WHERE a IS NULL;
212 }
213} {~/.*INDEX t2a1.*/}
214
215do_execsql_test index7-2.101 {
216 DROP INDEX t2a1;
217 UPDATE t2 SET a=b, b=b+10000;
218 SELECT b FROM t2 WHERE a=15;
219} {10015}
220do_execsql_test index7-2.102 {
221 CREATE INDEX t2a2 ON t2(a) WHERE a<100 OR a>200;
222 SELECT b FROM t2 WHERE a=15;
223 PRAGMA integrity_check;
224} {10015 ok}
225do_execsql_test index7-2.102eqp {
226 EXPLAIN QUERY PLAN
227 SELECT b FROM t2 WHERE a=15;
228} {~/.*INDEX t2a2.*/}
229do_execsql_test index7-2.103 {
230 SELECT b FROM t2 WHERE a=15 AND a<100;
231} {10015}
232do_execsql_test index7-2.103eqp {
233 EXPLAIN QUERY PLAN
234 SELECT b FROM t2 WHERE a=15 AND a<100;
235} {/.*INDEX t2a2.*/}
236do_execsql_test index7-2.104 {
237 SELECT b FROM t2 WHERE a=515 AND a>200;
238} {10515}
239do_execsql_test index7-2.104eqp {
240 EXPLAIN QUERY PLAN
241 SELECT b FROM t2 WHERE a=515 AND a>200;
242} {/.*INDEX t2a2.*/}
243
244# Partial UNIQUE indices
245#
246do_execsql_test index7-3.1 {
247 CREATE TABLE t3(a,b PRIMARY KEY) without rowid;
248 INSERT INTO t3 SELECT value, value FROM nums WHERE value<200;
249 UPDATE t3 SET a=999 WHERE b%5!=0;
250 CREATE UNIQUE INDEX t3a ON t3(a) WHERE a<>999;
251} {}
252do_test index7-3.2 {
253 # unable to insert a duplicate row a-value that is not 999.
254 catchsql {
255 INSERT INTO t3(a,b) VALUES(150, 'test1');
256 }
drhf9c8ce32013-11-05 13:33:55 +0000257} {1 {UNIQUE constraint failed: t3.a}}
drhd2694612013-11-04 22:04:17 +0000258do_test index7-3.3 {
259 # can insert multiple rows with a==999 because such rows are not
260 # part of the unique index.
261 catchsql {
262 INSERT INTO t3(a,b) VALUES(999, 'test1'), (999, 'test2');
263 }
264} {0 {}}
265do_execsql_test index7-3.4 {
266 SELECT count(*) FROM t3 WHERE a=999;
267} {162}
268integrity_check index7-3.5
269
270do_execsql_test index7-4.0 {
271 VACUUM;
272 PRAGMA integrity_check;
273} {ok}
274
275# Silently ignore database name qualifiers in partial indices.
276#
277do_execsql_test index7-5.0 {
278 CREATE INDEX t3b ON t3(b) WHERE xyzzy.t3.b BETWEEN 5 AND 10;
279 /* ^^^^^-- ignored */
280 ANALYZE;
281 SELECT count(*) FROM t3 WHERE t3.b BETWEEN 5 AND 10;
282 SELECT stat+0 FROM sqlite_stat1 WHERE idx='t3b';
283} {6 6}
284
dan54ae4e32014-08-27 17:48:15 +0000285# Verify that the problem identified by ticket [98d973b8f5] has been fixed.
dan08291692014-08-27 17:37:20 +0000286#
287do_execsql_test index7-6.1 {
288 CREATE TABLE t5(a, b);
289 CREATE TABLE t4(c, d);
290 INSERT INTO t5 VALUES(1, 'xyz');
291 INSERT INTO t4 VALUES('abc', 'not xyz');
292 SELECT * FROM (SELECT * FROM t5 WHERE a=1 AND b='xyz'), t4 WHERE c='abc';
293} {
294 1 xyz abc {not xyz}
295}
296do_execsql_test index7-6.2 {
297 CREATE INDEX i4 ON t4(c) WHERE d='xyz';
298 SELECT * FROM (SELECT * FROM t5 WHERE a=1 AND b='xyz'), t4 WHERE c='abc';
299} {
300 1 xyz abc {not xyz}
301}
302do_execsql_test index7-6.3 {
303 CREATE VIEW v4 AS SELECT * FROM t4;
304 INSERT INTO t4 VALUES('def', 'xyz');
305 SELECT * FROM v4 WHERE d='xyz' AND c='def'
306} {
307 def xyz
308}
309do_eqp_test index7-6.4 {
310 SELECT * FROM v4 WHERE d='xyz' AND c='def'
311} {
312 0 0 0 {SEARCH TABLE t4 USING INDEX i4 (c=?)}
313}
314
drhd2694612013-11-04 22:04:17 +0000315finish_test