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 focus
12 # of these tests is exclusive access mode (i.e. the thing activated by
13 # "PRAGMA locking_mode = EXCLUSIVE").
15 # $Id: exclusive.test,v 1.9 2008/09/24 14:03:43 danielk1977 Exp $
17 set testdir [file dirname $argv0]
18 source $testdir/tester.tcl
20 ifcapable {!pager_pragmas} {
25 file delete -force test2.db-journal
26 file delete -force test2.db
27 file delete -force test3.db-journal
28 file delete -force test3.db
29 file delete -force test4.db-journal
30 file delete -force test4.db
32 # The locking mode for the TEMP table is always "exclusive" for
33 # on-disk tables and "normal" for in-memory tables.
35 if {[info exists TEMP_STORE] && $TEMP_STORE>=2} {
38 set temp_mode exclusive
41 #----------------------------------------------------------------------
42 # Test cases exclusive-1.X test the PRAGMA logic.
44 do_test exclusive-1.0 {
47 pragma main.locking_mode;
48 pragma temp.locking_mode;
50 } [list normal normal $temp_mode]
51 do_test exclusive-1.1 {
53 pragma locking_mode = exclusive;
56 do_test exclusive-1.2 {
59 pragma main.locking_mode;
60 pragma temp.locking_mode;
62 } [list exclusive exclusive $temp_mode]
63 do_test exclusive-1.3 {
65 pragma locking_mode = normal;
68 do_test exclusive-1.4 {
71 pragma main.locking_mode;
72 pragma temp.locking_mode;
74 } [list normal normal $temp_mode]
75 do_test exclusive-1.5 {
77 pragma locking_mode = invalid;
80 do_test exclusive-1.6 {
83 pragma main.locking_mode;
84 pragma temp.locking_mode;
86 } [list normal normal $temp_mode]
88 do_test exclusive-1.7 {
90 pragma locking_mode = exclusive;
91 ATTACH 'test2.db' as aux;
94 pragma main.locking_mode;
95 pragma aux.locking_mode;
97 } {exclusive exclusive}
98 do_test exclusive-1.8 {
100 pragma main.locking_mode = normal;
103 pragma main.locking_mode;
104 pragma temp.locking_mode;
105 pragma aux.locking_mode;
107 } [list normal $temp_mode exclusive]
108 do_test exclusive-1.9 {
113 do_test exclusive-1.10 {
115 ATTACH 'test3.db' as aux2;
118 pragma main.locking_mode;
119 pragma aux.locking_mode;
120 pragma aux2.locking_mode;
122 } {normal exclusive exclusive}
123 do_test exclusive-1.11 {
125 pragma aux.locking_mode = normal;
128 pragma main.locking_mode;
129 pragma aux.locking_mode;
130 pragma aux2.locking_mode;
132 } {normal normal exclusive}
133 do_test exclusive-1.12 {
135 pragma locking_mode = normal;
138 pragma main.locking_mode;
139 pragma temp.locking_mode;
140 pragma aux.locking_mode;
141 pragma aux2.locking_mode;
143 } [list normal $temp_mode normal normal]
144 do_test exclusive-1.13 {
146 ATTACH 'test4.db' as aux3;
149 pragma main.locking_mode;
150 pragma temp.locking_mode;
151 pragma aux.locking_mode;
152 pragma aux2.locking_mode;
153 pragma aux3.locking_mode;
155 } [list normal $temp_mode normal normal normal]
157 do_test exclusive-1.99 {
166 #----------------------------------------------------------------------
167 # Test cases exclusive-2.X verify that connections in exclusive
168 # locking_mode do not relinquish locks.
170 do_test exclusive-2.0 {
172 CREATE TABLE abc(a, b, c);
173 INSERT INTO abc VALUES(1, 2, 3);
174 PRAGMA locking_mode = exclusive;
177 do_test exclusive-2.1 {
180 INSERT INTO abc VALUES(4, 5, 6);
184 do_test exclusive-2.2 {
185 # This causes connection 'db' (in exclusive mode) to establish
186 # a shared-lock on the db. The other connection should now be
187 # locked out as a writer.
192 do_test exclusive-2.4 {
197 do_test exclusive-2.5 {
199 INSERT INTO abc VALUES(7, 8, 9);
201 } {1 {database is locked}}
202 sqlite3_soft_heap_limit 0
203 do_test exclusive-2.6 {
204 # Because connection 'db' only has a shared-lock, the other connection
205 # will be able to get a RESERVED, but will fail to upgrade to EXCLUSIVE.
208 INSERT INTO abc VALUES(7, 8, 9);
213 } {1 {database is locked}}
214 do_test exclusive-2.7 {
218 } {1 {database is locked}}
219 do_test exclusive-2.8 {
224 sqlite3_soft_heap_limit $soft_limit
226 do_test exclusive-2.9 {
227 # Write the database to establish the exclusive lock with connection 'db.
229 INSERT INTO abc VALUES(7, 8, 9);
234 } {1 {database is locked}}
235 do_test exclusive-2.10 {
236 # Changing the locking-mode does not release any locks.
238 PRAGMA locking_mode = normal;
243 } {1 {database is locked}}
244 do_test exclusive-2.11 {
245 # After changing the locking mode, accessing the db releases locks.
252 } {1 2 3 4 5 6 7 8 9}
255 #----------------------------------------------------------------------
256 # Tests exclusive-3.X - test that a connection in exclusive mode
257 # truncates instead of deletes the journal file when committing
260 # These tests are not run on windows because the windows backend
261 # opens the journal file for exclusive access, preventing its contents
262 # from being inspected externally.
264 if {$tcl_platform(platform) != "windows"} {
265 proc filestate {fname} {
268 if {[file exists $fname]} {
270 set hdr [hexio_read $fname 0 28]
272 [expr {$hdr!="00000000000000000000000000000000000000000000000000000000"}]
274 list $exists $content
276 do_test exclusive-3.0 {
277 filestate test.db-journal
279 do_test exclusive-3.1 {
281 PRAGMA locking_mode = exclusive;
285 filestate test.db-journal
287 do_test exclusive-3.2 {
291 filestate test.db-journal
293 do_test exclusive-3.3 {
295 INSERT INTO abc VALUES('A', 'B', 'C');
299 do_test exclusive-3.4 {
302 UPDATE abc SET a = 1, b = 2, c = 3;
307 do_test exclusive-3.5 {
308 filestate test.db-journal
310 do_test exclusive-3.6 {
312 PRAGMA locking_mode = normal;
315 filestate test.db-journal
319 #----------------------------------------------------------------------
320 # Tests exclusive-4.X - test that rollback works correctly when
321 # in exclusive-access mode.
324 # The following procedure computes a "signature" for table "t3". If
325 # T3 changes in any way, the signature should change.
327 # This is used to test ROLLBACK. We gather a signature for t3, then
328 # make lots of changes to t3, then rollback and take another signature.
329 # The two signatures should be the same.
332 return [db eval {SELECT count(*), md5sum(x) FROM t3}]
335 do_test exclusive-4.0 {
336 execsql { PRAGMA locking_mode = exclusive; }
337 execsql { PRAGMA default_cache_size = 10; }
340 CREATE TABLE t3(x TEXT);
341 INSERT INTO t3 VALUES(randstr(10,400));
342 INSERT INTO t3 VALUES(randstr(10,400));
343 INSERT INTO t3 SELECT randstr(10,400) FROM t3;
344 INSERT INTO t3 SELECT randstr(10,400) FROM t3;
345 INSERT INTO t3 SELECT randstr(10,400) FROM t3;
346 INSERT INTO t3 SELECT randstr(10,400) FROM t3;
349 execsql {SELECT count(*) FROM t3;}
353 do_test exclusive-4.1 {
356 DELETE FROM t3 WHERE random()%10!=0;
357 INSERT INTO t3 SELECT randstr(10,10)||x FROM t3;
358 INSERT INTO t3 SELECT randstr(10,10)||x FROM t3;
359 SELECT count(*) FROM t3;
365 do_test exclusive-4.2 {
368 DELETE FROM t3 WHERE random()%10!=0;
369 INSERT INTO t3 SELECT randstr(10,10)||x FROM t3;
370 DELETE FROM t3 WHERE random()%10!=0;
371 INSERT INTO t3 SELECT randstr(10,10)||x FROM t3;
377 do_test exclusive-4.3 {
379 INSERT INTO t3 SELECT randstr(10,400) FROM t3 WHERE random()%10==0;
383 do_test exclusive-4.4 {
384 catch {set ::X [signature]}
386 do_test exclusive-4.5 {
388 PRAGMA locking_mode = NORMAL;
394 #----------------------------------------------------------------------
395 # Tests exclusive-5.X - test that statement journals are truncated
396 # instead of deleted when in exclusive access mode.
399 # Close and reopen the database so that the temp database is no
406 do_test exclusive-5.0 {
408 CREATE TABLE abc(a UNIQUE, b UNIQUE, c UNIQUE);
410 INSERT INTO abc VALUES(1, 2, 3);
411 INSERT INTO abc SELECT a+1, b+1, c+1 FROM abc;
414 do_test exclusive-5.1 {
415 # Three files are open: The db, journal and statement-journal.
416 set sqlite_open_file_count
418 do_test exclusive-5.2 {
422 # One file open: the db.
423 set sqlite_open_file_count
425 do_test exclusive-5.3 {
427 PRAGMA locking_mode = exclusive;
429 INSERT INTO abc VALUES(5, 6, 7);
431 # Two files open: the db and journal.
432 set sqlite_open_file_count
434 do_test exclusive-5.4 {
436 INSERT INTO abc SELECT a+10, b+10, c+10 FROM abc;
438 # Three files are open: The db, journal and statement-journal.
439 set sqlite_open_file_count
441 do_test exclusive-5.5 {
445 # Three files are still open: The db, journal and statement-journal.
446 set sqlite_open_file_count
448 do_test exclusive-5.6 {
450 PRAGMA locking_mode = normal;
453 } {normal 1 2 3 2 3 4 5 6 7 11 12 13 12 13 14 15 16 17}
454 do_test exclusive-5.7 {
456 set sqlite_open_file_count