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. The
12 # focus of this script is database locks.
14 # $Id: trans.test,v 1.38 2008/04/19 20:34:19 drh Exp $
17 set testdir [file dirname $argv0]
18 source $testdir/tester.tcl
21 # Create several tables to work with.
25 CREATE TABLE one(a int PRIMARY KEY, b text);
26 INSERT INTO one VALUES(1,'one');
27 INSERT INTO one VALUES(2,'two');
28 INSERT INTO one VALUES(3,'three');
29 SELECT b FROM one ORDER BY a;
32 integrity_check trans-1.0.1
35 CREATE TABLE two(a int PRIMARY KEY, b text);
36 INSERT INTO two VALUES(1,'I');
37 INSERT INTO two VALUES(5,'V');
38 INSERT INTO two VALUES(10,'X');
39 SELECT b FROM two ORDER BY a;
44 execsql {SELECT b FROM one ORDER BY a} altdb
47 execsql {SELECT b FROM two ORDER BY a} altdb
49 integrity_check trans-1.11
54 set v [catch {execsql {BEGIN}} msg]
58 set v [catch {execsql {END}} msg]
62 set v [catch {execsql {BEGIN TRANSACTION}} msg]
66 set v [catch {execsql {COMMIT TRANSACTION}} msg]
70 set v [catch {execsql {BEGIN TRANSACTION 'foo'}} msg]
74 set v [catch {execsql {ROLLBACK TRANSACTION 'foo'}} msg]
80 SELECT a FROM one ORDER BY a;
81 SELECT a FROM two ORDER BY a;
85 integrity_check trans-2.11
87 # Check the locking behavior
92 UPDATE one SET a = 0 WHERE 0;
93 SELECT a FROM one ORDER BY a;
98 SELECT a FROM two ORDER BY a;
104 SELECT a FROM one ORDER BY a;
109 INSERT INTO one VALUES(4,'four');
114 SELECT a FROM two ORDER BY a;
119 SELECT a FROM one ORDER BY a;
124 INSERT INTO two VALUES(4,'IV');
129 SELECT a FROM two ORDER BY a;
134 SELECT a FROM one ORDER BY a;
138 execsql {END TRANSACTION}
142 set v [catch {execsql {
143 SELECT a FROM two ORDER BY a;
148 set v [catch {execsql {
149 SELECT a FROM one ORDER BY a;
154 set v [catch {execsql {
155 SELECT a FROM two ORDER BY a;
160 set v [catch {execsql {
161 SELECT a FROM one ORDER BY a;
165 integrity_check trans-3.15
168 set v [catch {execsql {
172 } {1 {cannot commit - no transaction is active}}
174 set v [catch {execsql {
178 } {1 {cannot rollback - no transaction is active}}
182 UPDATE two SET a = 0 WHERE 0;
183 SELECT a FROM two ORDER BY a;
188 SELECT a FROM two ORDER BY a;
193 SELECT a FROM one ORDER BY a;
199 SELECT a FROM one ORDER BY a;
201 } {1 {cannot start a transaction within a transaction}}
204 SELECT a FROM two ORDER BY a;
209 SELECT a FROM one ORDER BY a;
213 set v [catch {execsql {
215 SELECT a FROM two ORDER BY a;
220 set v [catch {execsql {
221 SELECT a FROM two ORDER BY a;
226 set v [catch {execsql {
227 SELECT a FROM one ORDER BY a;
231 integrity_check trans-4.12
239 integrity_check trans-4.99
241 # Check out the commit/rollback behavior of the database
244 execsql {SELECT name FROM sqlite_master WHERE type='table' ORDER BY name}
247 execsql {BEGIN TRANSACTION}
248 execsql {SELECT name FROM sqlite_master WHERE type='table' ORDER BY name}
251 execsql {CREATE TABLE one(a text, b int)}
252 execsql {SELECT name FROM sqlite_master WHERE type='table' ORDER BY name}
255 execsql {SELECT a,b FROM one ORDER BY b}
258 execsql {INSERT INTO one(a,b) VALUES('hello', 1)}
259 execsql {SELECT a,b FROM one ORDER BY b}
263 execsql {SELECT name FROM sqlite_master WHERE type='table' ORDER BY name}
267 execsql {SELECT a,b FROM one ORDER BY b}
270 } {1 {no such table: one}}
272 # Test commits and rollbacks of table CREATE TABLEs, CREATE INDEXs
273 # DROP TABLEs and DROP INDEXs
277 SELECT name fROM sqlite_master
278 WHERE type='table' OR type='index'
285 CREATE TABLE t1(a int, b int, c int);
286 SELECT name fROM sqlite_master
287 WHERE type='table' OR type='index'
293 CREATE INDEX i1 ON t1(a);
294 SELECT name fROM sqlite_master
295 WHERE type='table' OR type='index'
302 SELECT name fROM sqlite_master
303 WHERE type='table' OR type='index'
310 CREATE TABLE t2(a int, b int, c int);
311 CREATE INDEX i2a ON t2(a);
312 CREATE INDEX i2b ON t2(b);
314 SELECT name fROM sqlite_master
315 WHERE type='table' OR type='index'
322 SELECT name fROM sqlite_master
323 WHERE type='table' OR type='index'
331 SELECT name fROM sqlite_master
332 WHERE type='table' OR type='index'
339 SELECT name fROM sqlite_master
340 WHERE type='table' OR type='index'
348 CREATE TABLE t2(x int, y int, z int);
349 CREATE INDEX i2x ON t2(x);
350 CREATE INDEX i2y ON t2(y);
351 INSERT INTO t2 VALUES(1,2,3);
352 SELECT name fROM sqlite_master
353 WHERE type='table' OR type='index'
360 SELECT name fROM sqlite_master
361 WHERE type='table' OR type='index'
372 SELECT x FROM t2 WHERE y=2;
380 SELECT name fROM sqlite_master
381 WHERE type='table' OR type='index'
386 set r [catch {execsql {
390 } {1 {no such table: t2}}
394 SELECT name fROM sqlite_master
395 WHERE type='table' OR type='index'
404 integrity_check trans-5.23
407 # Try to DROP and CREATE tables and indices with the same name
408 # within a transaction. Make sure ROLLBACK works.
412 INSERT INTO t1 VALUES(1,2,3);
415 CREATE TABLE t1(p,q,r);
422 INSERT INTO t1 VALUES(1,2,3);
425 CREATE TABLE t1(p,q,r);
432 INSERT INTO t1 VALUES(1,2,3);
440 CREATE TABLE t1(a,b,c);
441 INSERT INTO t1 VALUES(4,5,6);
456 CREATE TABLE t1(a,b,c);
457 INSERT INTO t1 VALUES(4,5,6);
467 } {1 {no such table: t1}}
469 # Repeat on a table with an automatically generated index.
473 CREATE TABLE t1(a unique,b,c);
474 INSERT INTO t1 VALUES(1,2,3);
477 CREATE TABLE t1(p unique,q,r);
486 CREATE TABLE t1(p unique,q,r);
493 INSERT INTO t1 VALUES(1,2,3);
501 CREATE TABLE t1(a unique,b,c);
502 INSERT INTO t1 VALUES(4,5,6);
517 CREATE TABLE t1(a unique,b,c);
518 INSERT INTO t1 VALUES(4,5,6);
528 } {1 {no such table: t1}}
532 CREATE TABLE t1(a integer primary key,b,c);
533 INSERT INTO t1 VALUES(1,-2,-3);
534 INSERT INTO t1 VALUES(4,-5,-6);
540 CREATE INDEX i1 ON t1(b);
541 SELECT * FROM t1 WHERE b<1;
548 SELECT * FROM t1 WHERE b<1;
554 SELECT * FROM t1 WHERE b<1;
562 SELECT * FROM t1 WHERE b<1;
570 CREATE INDEX i1 ON t1(c);
571 SELECT * FROM t1 WHERE b<1;
576 SELECT * FROM t1 WHERE c<1;
582 SELECT * FROM t1 WHERE b<1;
587 SELECT * FROM t1 WHERE c<1;
591 # The following repeats steps 6.20 through 6.28, but puts a "unique"
592 # constraint the first field of the table in order to generate an
599 CREATE TABLE t1(a int unique,b,c);
601 INSERT INTO t1 VALUES(1,-2,-3);
602 INSERT INTO t1 VALUES(4,-5,-6);
603 SELECT * FROM t1 ORDER BY a;
608 CREATE INDEX i1 ON t1(b);
609 SELECT * FROM t1 WHERE b<1;
616 SELECT * FROM t1 WHERE b<1;
622 SELECT * FROM t1 WHERE b<1;
630 SELECT * FROM t1 WHERE b<1;
638 CREATE INDEX i1 ON t1(c);
639 SELECT * FROM t1 WHERE b<1;
644 SELECT * FROM t1 WHERE c<1;
650 SELECT * FROM t1 WHERE c<1;
656 SELECT * FROM t1 WHERE b<1;
661 SELECT * FROM t1 WHERE c<1;
664 integrity_check trans-6.40
666 # Test to make sure rollback restores the database back to its original
671 for {set i 0} {$i<1000} {incr i} {
672 set r1 [expr {rand()}]
673 set r2 [expr {rand()}]
674 set r3 [expr {rand()}]
675 execsql "INSERT INTO t2 VALUES($r1,$r2,$r3)"
678 set ::checksum [execsql {SELECT md5sum(x,y,z) FROM t2}]
680 execsql {SELECT md5sum(type,name,tbl_name,rootpage,sql) FROM sqlite_master}
682 execsql {SELECT count(*) FROM t2}
685 execsql {SELECT md5sum(x,y,z) FROM t2}
687 do_test trans-7.2.1 {
688 execsql {SELECT md5sum(type,name,tbl_name,rootpage,sql) FROM sqlite_master}
695 SELECT md5sum(x,y,z) FROM t2;
701 INSERT INTO t2 SELECT * FROM t2;
703 SELECT md5sum(x,y,z) FROM t2;
711 SELECT md5sum(x,y,z) FROM t2;
717 INSERT INTO t2 SELECT * FROM t2;
719 SELECT md5sum(x,y,z) FROM t2;
725 CREATE TABLE t3 AS SELECT * FROM t2;
726 INSERT INTO t2 SELECT * FROM t3;
728 SELECT md5sum(x,y,z) FROM t2;
732 execsql {SELECT md5sum(type,name,tbl_name,rootpage,sql) FROM sqlite_master}
738 CREATE TEMP TABLE t3 AS SELECT * FROM t2;
739 INSERT INTO t2 SELECT * FROM t3;
741 SELECT md5sum(x,y,z) FROM t2;
746 execsql {SELECT md5sum(type,name,tbl_name,rootpage,sql) FROM sqlite_master}
752 CREATE TEMP TABLE t3 AS SELECT * FROM t2;
753 INSERT INTO t2 SELECT * FROM t3;
756 CREATE INDEX i3a ON t3(x);
758 SELECT md5sum(x,y,z) FROM t2;
763 execsql {SELECT md5sum(type,name,tbl_name,rootpage,sql) FROM sqlite_master}
771 SELECT md5sum(x,y,z) FROM t2;
776 execsql {SELECT md5sum(type,name,tbl_name,rootpage,sql) FROM sqlite_master}
778 integrity_check trans-7.15
780 # Arrange for another process to begin modifying the database but abort
781 # and die in the middle of the modification. Then have this process read
782 # the database. This process should detect the journal file and roll it
783 # back. Verify that this happens correctly.
785 print_text "TclSqlite3-2" "begin"
786 set fd [open test.tcl w]
790 PRAGMA default_cache_size=20;
792 CREATE TABLE t3 AS SELECT * FROM t2;
798 print_text "TclSqlite3-2" "end"
801 catch {exec [info nameofexec] test.tcl}
802 execsql {SELECT md5sum(x,y,z) FROM t2}
805 execsql {SELECT md5sum(type,name,tbl_name,rootpage,sql) FROM sqlite_master}
807 integrity_check trans-8.3
809 print_text "TclSqlite3-2" "begin"
810 set fd [open test.tcl w]
814 PRAGMA journal_mode=persist;
815 PRAGMA default_cache_size=20;
817 CREATE TABLE t3 AS SELECT * FROM t2;
823 print_text "TclSqlite3-2" "end"
826 catch {exec [info nameofexec] test.tcl}
827 execsql {SELECT md5sum(x,y,z) FROM t2}
830 execsql {SELECT md5sum(type,name,tbl_name,rootpage,sql) FROM sqlite_master}
832 integrity_check trans-8.6
834 # In the following sequence of tests, compute the MD5 sum of the content
835 # of a table, make lots of modifications to that table, then do a rollback.
836 # Verify that after the rollback, the MD5 checksum is unchanged.
840 PRAGMA default_cache_size=10;
846 CREATE TABLE t3(x TEXT);
847 INSERT INTO t3 VALUES(randstr(10,400));
848 INSERT INTO t3 VALUES(randstr(10,400));
849 INSERT INTO t3 SELECT randstr(10,400) FROM t3;
850 INSERT INTO t3 SELECT randstr(10,400) FROM t3;
851 INSERT INTO t3 SELECT randstr(10,400) FROM t3;
852 INSERT INTO t3 SELECT randstr(10,400) FROM t3;
853 INSERT INTO t3 SELECT randstr(10,400) FROM t3;
854 INSERT INTO t3 SELECT randstr(10,400) FROM t3;
855 INSERT INTO t3 SELECT randstr(10,400) FROM t3;
856 INSERT INTO t3 SELECT randstr(10,400) FROM t3;
857 INSERT INTO t3 SELECT randstr(10,400) FROM t3;
859 SELECT count(*) FROM t3;
863 # The following procedure computes a "signature" for table "t3". If
864 # T3 changes in any way, the signature should change.
866 # This is used to test ROLLBACK. We gather a signature for t3, then
867 # make lots of changes to t3, then rollback and take another signature.
868 # The two signatures should be the same.
871 return [db eval {SELECT count(*), md5sum(x) FROM t3}]
874 # Repeat the following group of tests 20 times for quick testing and
875 # 40 times for full testing. Each iteration of the test makes table
876 # t3 a little larger, and thus takes a little longer, so doing 40 tests
877 # is more than 2.0 times slower than doing 20 tests. Considerably more.
879 if {[info exists ISQUICK]} {
881 } elseif {[info exists SOAKTEST]} {
887 # Do rollbacks. Make sure the signature does not change.
889 for {set i 2} {$i<=$limit} {incr i} {
890 set ::sig [signature]
891 set cnt [lindex $::sig 0]
893 execsql {PRAGMA fullfsync=ON}
895 execsql {PRAGMA fullfsync=OFF}
897 set sqlite_sync_count 0
898 set sqlite_fullsync_count 0
899 do_test trans-9.$i.1-$cnt {
902 DELETE FROM t3 WHERE random()%10!=0;
903 INSERT INTO t3 SELECT randstr(10,10)||x FROM t3;
904 INSERT INTO t3 SELECT randstr(10,10)||x FROM t3;
909 do_test trans-9.$i.2-$cnt {
912 DELETE FROM t3 WHERE random()%10!=0;
913 INSERT INTO t3 SELECT randstr(10,10)||x FROM t3;
914 DELETE FROM t3 WHERE random()%10!=0;
915 INSERT INTO t3 SELECT randstr(10,10)||x FROM t3;
921 do_test trans-9.$i.3-$cnt {
923 INSERT INTO t3 SELECT randstr(10,400) FROM t3 WHERE random()%10==0;
926 if {$tcl_platform(platform)=="unix"} {
927 do_test trans-9.$i.4-$cnt {
928 expr {$sqlite_sync_count>0}
930 ifcapable pager_pragmas {
931 do_test trans-9.$i.5-$cnt {
932 expr {$sqlite_fullsync_count>0}
935 do_test trans-9.$i.5-$cnt {
936 expr {$sqlite_fullsync_count==0}
941 set ::pager_old_format 0