First public contribution.
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 #***********************************************************************
11 # This file implements regression tests for SQLite library. This
12 # file is a copy of "trans.test" modified to run under autovacuum mode.
13 # the point is to stress the autovacuum logic and try to get it to fail.
15 # $Id: avtrans.test,v 1.6 2007/09/12 17:01:45 danielk1977 Exp $
18 set testdir [file dirname $argv0]
19 source $testdir/tester.tcl
22 # Create several tables to work with.
26 PRAGMA auto_vacuum=ON;
27 CREATE TABLE one(a int PRIMARY KEY, b text);
28 INSERT INTO one VALUES(1,'one');
29 INSERT INTO one VALUES(2,'two');
30 INSERT INTO one VALUES(3,'three');
31 SELECT b FROM one ORDER BY a;
36 CREATE TABLE two(a int PRIMARY KEY, b text);
37 INSERT INTO two VALUES(1,'I');
38 INSERT INTO two VALUES(5,'V');
39 INSERT INTO two VALUES(10,'X');
40 SELECT b FROM two ORDER BY a;
45 execsql {SELECT b FROM one ORDER BY a} altdb
47 do_test avtrans-1.10 {
48 execsql {SELECT b FROM two ORDER BY a} altdb
50 integrity_check avtrans-1.11
55 set v [catch {execsql {BEGIN}} msg]
59 set v [catch {execsql {END}} msg]
63 set v [catch {execsql {BEGIN TRANSACTION}} msg]
67 set v [catch {execsql {COMMIT TRANSACTION}} msg]
71 set v [catch {execsql {BEGIN TRANSACTION 'foo'}} msg]
75 set v [catch {execsql {ROLLBACK TRANSACTION 'foo'}} msg]
78 do_test avtrans-2.10 {
81 SELECT a FROM one ORDER BY a;
82 SELECT a FROM two ORDER BY a;
86 integrity_check avtrans-2.11
88 # Check the locking behavior
90 sqlite3_soft_heap_limit 0
94 UPDATE one SET a = 0 WHERE 0;
95 SELECT a FROM one ORDER BY a;
100 SELECT a FROM two ORDER BY a;
103 do_test avtrans-3.3 {
105 SELECT a FROM one ORDER BY a;
108 do_test avtrans-3.4 {
110 INSERT INTO one VALUES(4,'four');
113 do_test avtrans-3.5 {
115 SELECT a FROM two ORDER BY a;
118 do_test avtrans-3.6 {
120 SELECT a FROM one ORDER BY a;
123 do_test avtrans-3.7 {
125 INSERT INTO two VALUES(4,'IV');
128 do_test avtrans-3.8 {
130 SELECT a FROM two ORDER BY a;
133 do_test avtrans-3.9 {
135 SELECT a FROM one ORDER BY a;
138 do_test avtrans-3.10 {
139 execsql {END TRANSACTION}
141 do_test avtrans-3.11 {
142 set v [catch {execsql {
143 SELECT a FROM two ORDER BY a;
147 do_test avtrans-3.12 {
148 set v [catch {execsql {
149 SELECT a FROM one ORDER BY a;
153 do_test avtrans-3.13 {
154 set v [catch {execsql {
155 SELECT a FROM two ORDER BY a;
159 do_test avtrans-3.14 {
160 set v [catch {execsql {
161 SELECT a FROM one ORDER BY a;
165 sqlite3_soft_heap_limit $soft_limit
166 integrity_check avtrans-3.15
168 do_test avtrans-4.1 {
169 set v [catch {execsql {
173 } {1 {cannot commit - no transaction is active}}
174 do_test avtrans-4.2 {
175 set v [catch {execsql {
179 } {1 {cannot rollback - no transaction is active}}
180 do_test avtrans-4.3 {
183 UPDATE two SET a = 0 WHERE 0;
184 SELECT a FROM two ORDER BY a;
187 do_test avtrans-4.4 {
189 SELECT a FROM two ORDER BY a;
192 do_test avtrans-4.5 {
194 SELECT a FROM one ORDER BY a;
197 do_test avtrans-4.6 {
200 SELECT a FROM one ORDER BY a;
202 } {1 {cannot start a transaction within a transaction}}
203 do_test avtrans-4.7 {
205 SELECT a FROM two ORDER BY a;
208 do_test avtrans-4.8 {
210 SELECT a FROM one ORDER BY a;
213 do_test avtrans-4.9 {
214 set v [catch {execsql {
216 SELECT a FROM two ORDER BY a;
220 do_test avtrans-4.10 {
221 set v [catch {execsql {
222 SELECT a FROM two ORDER BY a;
226 do_test avtrans-4.11 {
227 set v [catch {execsql {
228 SELECT a FROM one ORDER BY a;
232 integrity_check avtrans-4.12
233 do_test avtrans-4.98 {
240 integrity_check avtrans-4.99
242 # Check out the commit/rollback behavior of the database
244 do_test avtrans-5.1 {
245 execsql {SELECT name FROM sqlite_master WHERE type='table' ORDER BY name}
247 do_test avtrans-5.2 {
248 execsql {BEGIN TRANSACTION}
249 execsql {SELECT name FROM sqlite_master WHERE type='table' ORDER BY name}
251 do_test avtrans-5.3 {
252 execsql {CREATE TABLE one(a text, b int)}
253 execsql {SELECT name FROM sqlite_master WHERE type='table' ORDER BY name}
255 do_test avtrans-5.4 {
256 execsql {SELECT a,b FROM one ORDER BY b}
258 do_test avtrans-5.5 {
259 execsql {INSERT INTO one(a,b) VALUES('hello', 1)}
260 execsql {SELECT a,b FROM one ORDER BY b}
262 do_test avtrans-5.6 {
264 execsql {SELECT name FROM sqlite_master WHERE type='table' ORDER BY name}
266 do_test avtrans-5.7 {
268 execsql {SELECT a,b FROM one ORDER BY b}
271 } {1 {no such table: one}}
273 # Test commits and rollbacks of table CREATE TABLEs, CREATE INDEXs
274 # DROP TABLEs and DROP INDEXs
276 do_test avtrans-5.8 {
278 SELECT name fROM sqlite_master
279 WHERE type='table' OR type='index'
283 do_test avtrans-5.9 {
286 CREATE TABLE t1(a int, b int, c int);
287 SELECT name fROM sqlite_master
288 WHERE type='table' OR type='index'
292 do_test avtrans-5.10 {
294 CREATE INDEX i1 ON t1(a);
295 SELECT name fROM sqlite_master
296 WHERE type='table' OR type='index'
300 do_test avtrans-5.11 {
303 SELECT name fROM sqlite_master
304 WHERE type='table' OR type='index'
308 do_test avtrans-5.12 {
311 CREATE TABLE t2(a int, b int, c int);
312 CREATE INDEX i2a ON t2(a);
313 CREATE INDEX i2b ON t2(b);
315 SELECT name fROM sqlite_master
316 WHERE type='table' OR type='index'
320 do_test avtrans-5.13 {
323 SELECT name fROM sqlite_master
324 WHERE type='table' OR type='index'
328 do_test avtrans-5.14 {
332 SELECT name fROM sqlite_master
333 WHERE type='table' OR type='index'
337 do_test avtrans-5.15 {
340 SELECT name fROM sqlite_master
341 WHERE type='table' OR type='index'
345 do_test avtrans-5.16 {
349 CREATE TABLE t2(x int, y int, z int);
350 CREATE INDEX i2x ON t2(x);
351 CREATE INDEX i2y ON t2(y);
352 INSERT INTO t2 VALUES(1,2,3);
353 SELECT name fROM sqlite_master
354 WHERE type='table' OR type='index'
358 do_test avtrans-5.17 {
361 SELECT name fROM sqlite_master
362 WHERE type='table' OR type='index'
366 do_test avtrans-5.18 {
371 do_test avtrans-5.19 {
373 SELECT x FROM t2 WHERE y=2;
376 do_test avtrans-5.20 {
381 SELECT name fROM sqlite_master
382 WHERE type='table' OR type='index'
386 do_test avtrans-5.21 {
387 set r [catch {execsql {
391 } {1 {no such table: t2}}
392 do_test avtrans-5.22 {
395 SELECT name fROM sqlite_master
396 WHERE type='table' OR type='index'
400 do_test avtrans-5.23 {
405 integrity_check avtrans-5.23
408 # Try to DROP and CREATE tables and indices with the same name
409 # within a transaction. Make sure ROLLBACK works.
411 do_test avtrans-6.1 {
413 INSERT INTO t1 VALUES(1,2,3);
416 CREATE TABLE t1(p,q,r);
421 do_test avtrans-6.2 {
423 INSERT INTO t1 VALUES(1,2,3);
426 CREATE TABLE t1(p,q,r);
431 do_test avtrans-6.3 {
433 INSERT INTO t1 VALUES(1,2,3);
437 do_test avtrans-6.4 {
441 CREATE TABLE t1(a,b,c);
442 INSERT INTO t1 VALUES(4,5,6);
447 do_test avtrans-6.5 {
453 do_test avtrans-6.6 {
457 CREATE TABLE t1(a,b,c);
458 INSERT INTO t1 VALUES(4,5,6);
463 do_test avtrans-6.7 {
468 } {1 {no such table: t1}}
470 # Repeat on a table with an automatically generated index.
472 do_test avtrans-6.10 {
474 CREATE TABLE t1(a unique,b,c);
475 INSERT INTO t1 VALUES(1,2,3);
478 CREATE TABLE t1(p unique,q,r);
483 do_test avtrans-6.11 {
487 CREATE TABLE t1(p unique,q,r);
492 do_test avtrans-6.12 {
494 INSERT INTO t1 VALUES(1,2,3);
498 do_test avtrans-6.13 {
502 CREATE TABLE t1(a unique,b,c);
503 INSERT INTO t1 VALUES(4,5,6);
508 do_test avtrans-6.14 {
514 do_test avtrans-6.15 {
518 CREATE TABLE t1(a unique,b,c);
519 INSERT INTO t1 VALUES(4,5,6);
524 do_test avtrans-6.16 {
529 } {1 {no such table: t1}}
531 do_test avtrans-6.20 {
533 CREATE TABLE t1(a integer primary key,b,c);
534 INSERT INTO t1 VALUES(1,-2,-3);
535 INSERT INTO t1 VALUES(4,-5,-6);
539 do_test avtrans-6.21 {
541 CREATE INDEX i1 ON t1(b);
542 SELECT * FROM t1 WHERE b<1;
545 do_test avtrans-6.22 {
549 SELECT * FROM t1 WHERE b<1;
553 do_test avtrans-6.23 {
555 SELECT * FROM t1 WHERE b<1;
558 do_test avtrans-6.24 {
563 SELECT * FROM t1 WHERE b<1;
567 do_test avtrans-6.25 {
571 CREATE INDEX i1 ON t1(c);
572 SELECT * FROM t1 WHERE b<1;
575 do_test avtrans-6.26 {
577 SELECT * FROM t1 WHERE c<1;
580 do_test avtrans-6.27 {
583 SELECT * FROM t1 WHERE b<1;
586 do_test avtrans-6.28 {
588 SELECT * FROM t1 WHERE c<1;
592 # The following repeats steps 6.20 through 6.28, but puts a "unique"
593 # constraint the first field of the table in order to generate an
596 do_test avtrans-6.30 {
600 CREATE TABLE t1(a int unique,b,c);
602 INSERT INTO t1 VALUES(1,-2,-3);
603 INSERT INTO t1 VALUES(4,-5,-6);
604 SELECT * FROM t1 ORDER BY a;
607 do_test avtrans-6.31 {
609 CREATE INDEX i1 ON t1(b);
610 SELECT * FROM t1 WHERE b<1;
613 do_test avtrans-6.32 {
617 SELECT * FROM t1 WHERE b<1;
621 do_test avtrans-6.33 {
623 SELECT * FROM t1 WHERE b<1;
626 do_test avtrans-6.34 {
631 SELECT * FROM t1 WHERE b<1;
635 do_test avtrans-6.35 {
639 CREATE INDEX i1 ON t1(c);
640 SELECT * FROM t1 WHERE b<1;
643 do_test avtrans-6.36 {
645 SELECT * FROM t1 WHERE c<1;
648 do_test avtrans-6.37 {
651 SELECT * FROM t1 WHERE c<1;
654 do_test avtrans-6.38 {
657 SELECT * FROM t1 WHERE b<1;
660 do_test avtrans-6.39 {
662 SELECT * FROM t1 WHERE c<1;
665 integrity_check avtrans-6.40
667 ifcapable !floatingpoint {
672 # Test to make sure rollback restores the database back to its original
675 do_test avtrans-7.1 {
677 for {set i 0} {$i<1000} {incr i} {
678 set r1 [expr {rand()}]
679 set r2 [expr {rand()}]
680 set r3 [expr {rand()}]
681 execsql "INSERT INTO t2 VALUES($r1,$r2,$r3)"
684 set ::checksum [execsql {SELECT md5sum(x,y,z) FROM t2}]
686 execsql {SELECT md5sum(type,name,tbl_name,rootpage,sql) FROM sqlite_master}
688 execsql {SELECT count(*) FROM t2}
690 do_test avtrans-7.2 {
691 execsql {SELECT md5sum(x,y,z) FROM t2}
693 do_test avtrans-7.2.1 {
694 execsql {SELECT md5sum(type,name,tbl_name,rootpage,sql) FROM sqlite_master}
696 do_test avtrans-7.3 {
701 SELECT md5sum(x,y,z) FROM t2;
704 do_test avtrans-7.4 {
707 INSERT INTO t2 SELECT * FROM t2;
709 SELECT md5sum(x,y,z) FROM t2;
712 do_test avtrans-7.5 {
717 SELECT md5sum(x,y,z) FROM t2;
720 do_test avtrans-7.6 {
723 INSERT INTO t2 SELECT * FROM t2;
725 SELECT md5sum(x,y,z) FROM t2;
728 do_test avtrans-7.7 {
731 CREATE TABLE t3 AS SELECT * FROM t2;
732 INSERT INTO t2 SELECT * FROM t3;
734 SELECT md5sum(x,y,z) FROM t2;
737 do_test avtrans-7.8 {
738 execsql {SELECT md5sum(type,name,tbl_name,rootpage,sql) FROM sqlite_master}
741 do_test avtrans-7.9 {
744 CREATE TEMP TABLE t3 AS SELECT * FROM t2;
745 INSERT INTO t2 SELECT * FROM t3;
747 SELECT md5sum(x,y,z) FROM t2;
751 do_test avtrans-7.10 {
752 execsql {SELECT md5sum(type,name,tbl_name,rootpage,sql) FROM sqlite_master}
755 do_test avtrans-7.11 {
758 CREATE TEMP TABLE t3 AS SELECT * FROM t2;
759 INSERT INTO t2 SELECT * FROM t3;
762 CREATE INDEX i3a ON t3(x);
764 SELECT md5sum(x,y,z) FROM t2;
768 do_test avtrans-7.12 {
769 execsql {SELECT md5sum(type,name,tbl_name,rootpage,sql) FROM sqlite_master}
772 do_test avtrans-7.13 {
777 SELECT md5sum(x,y,z) FROM t2;
781 do_test avtrans-7.14 {
782 execsql {SELECT md5sum(type,name,tbl_name,rootpage,sql) FROM sqlite_master}
784 integrity_check avtrans-7.15
786 # Arrange for another process to begin modifying the database but abort
787 # and die in the middle of the modification. Then have this process read
788 # the database. This process should detect the journal file and roll it
789 # back. Verify that this happens correctly.
791 set fd [open test.tcl w]
795 PRAGMA default_cache_size=20;
797 CREATE TABLE t3 AS SELECT * FROM t2;
803 do_test avtrans-8.1 {
804 catch {exec [info nameofexec] test.tcl}
805 execsql {SELECT md5sum(x,y,z) FROM t2}
807 do_test avtrans-8.2 {
808 execsql {SELECT md5sum(type,name,tbl_name,rootpage,sql) FROM sqlite_master}
810 integrity_check avtrans-8.3
812 # In the following sequence of tests, compute the MD5 sum of the content
813 # of a table, make lots of modifications to that table, then do a rollback.
814 # Verify that after the rollback, the MD5 checksum is unchanged.
816 do_test avtrans-9.1 {
818 PRAGMA default_cache_size=10;
824 CREATE TABLE t3(x TEXT);
825 INSERT INTO t3 VALUES(randstr(10,400));
826 INSERT INTO t3 VALUES(randstr(10,400));
827 INSERT INTO t3 SELECT randstr(10,400) FROM t3;
828 INSERT INTO t3 SELECT randstr(10,400) FROM t3;
829 INSERT INTO t3 SELECT randstr(10,400) FROM t3;
830 INSERT INTO t3 SELECT randstr(10,400) FROM t3;
831 INSERT INTO t3 SELECT randstr(10,400) FROM t3;
832 INSERT INTO t3 SELECT randstr(10,400) FROM t3;
833 INSERT INTO t3 SELECT randstr(10,400) FROM t3;
834 INSERT INTO t3 SELECT randstr(10,400) FROM t3;
835 INSERT INTO t3 SELECT randstr(10,400) FROM t3;
837 SELECT count(*) FROM t3;
841 # The following procedure computes a "signature" for table "t3". If
842 # T3 changes in any way, the signature should change.
844 # This is used to test ROLLBACK. We gather a signature for t3, then
845 # make lots of changes to t3, then rollback and take another signature.
846 # The two signatures should be the same.
849 return [db eval {SELECT count(*), md5sum(x) FROM t3}]
852 # Repeat the following group of tests 20 times for quick testing and
853 # 40 times for full testing. Each iteration of the test makes table
854 # t3 a little larger, and thus takes a little longer, so doing 40 tests
855 # is more than 2.0 times slower than doing 20 tests. Considerably more.
857 if {[info exists ISQUICK]} {
863 # Do rollbacks. Make sure the signature does not change.
865 for {set i 2} {$i<=$limit} {incr i} {
866 set ::sig [signature]
867 set cnt [lindex $::sig 0]
869 execsql {PRAGMA fullfsync=ON}
871 execsql {PRAGMA fullfsync=OFF}
873 set sqlite_sync_count 0
874 set sqlite_fullsync_count 0
875 do_test avtrans-9.$i.1-$cnt {
878 DELETE FROM t3 WHERE random()%10!=0;
879 INSERT INTO t3 SELECT randstr(10,10)||x FROM t3;
880 INSERT INTO t3 SELECT randstr(10,10)||x FROM t3;
885 do_test avtrans-9.$i.2-$cnt {
888 DELETE FROM t3 WHERE random()%10!=0;
889 INSERT INTO t3 SELECT randstr(10,10)||x FROM t3;
890 DELETE FROM t3 WHERE random()%10!=0;
891 INSERT INTO t3 SELECT randstr(10,10)||x FROM t3;
897 do_test avtrans-9.$i.3-$cnt {
899 INSERT INTO t3 SELECT randstr(10,400) FROM t3 WHERE random()%10==0;
902 if {$tcl_platform(platform)=="unix"} {
903 do_test avtrans-9.$i.4-$cnt {
904 expr {$sqlite_sync_count>0}
906 ifcapable pager_pragmas {
907 do_test avtrans-9.$i.5-$cnt {
908 expr {$sqlite_fullsync_count>0}
911 do_test avtrans-9.$i.5-$cnt {
912 expr {$sqlite_fullsync_count==0}
917 set ::pager_old_format 0
919 integrity_check avtrans-10.1