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.
13 # $Id: exclusive2.test,v 1.9 2008/08/22 00:25:53 aswift Exp $
15 set testdir [file dirname $argv0]
16 source $testdir/tester.tcl
18 ifcapable {!pager_pragmas} {
23 # This module does not work right if the cache spills at unexpected
24 # moments. So disable the soft-heap-limit.
26 sqlite3_soft_heap_limit 0
28 proc pagerChangeCounter {filename new {fd ""}} {
30 set fd [open $filename RDWR]
31 fconfigure $fd -translation binary -encoding binary
38 set a [expr {($new&0xFF000000)>>24}]
39 set b [expr {($new&0x00FF0000)>>16}]
40 set c [expr {($new&0x0000FF00)>>8}]
41 set d [expr {($new&0x000000FF)}]
42 puts -nonewline $fd [binary format cccc $a $b $c $d]
47 foreach {a b c d} [list 0 0 0 0] {}
48 binary scan [read $fd 4] cccc a b c d
49 set ret [expr ($a&0x000000FF)<<24]
50 incr ret [expr ($b&0x000000FF)<<16]
51 incr ret [expr ($c&0x000000FF)<<8]
52 incr ret [expr ($d&0x000000FF)<<0]
54 if {$needClose} {close $fd}
58 proc readPagerChangeCounter {filename} {
59 set fd [open $filename RDONLY]
60 fconfigure $fd -translation binary -encoding binary
63 foreach {a b c d} [list 0 0 0 0] {}
64 binary scan [read $fd 4] cccc a b c d
65 set ret [expr ($a&0x000000FF)<<24]
66 incr ret [expr ($b&0x000000FF)<<16]
67 incr ret [expr ($c&0x000000FF)<<8]
68 incr ret [expr ($d&0x000000FF)<<0]
75 proc t1sig {{db db}} {
76 execsql {SELECT count(*), md5sum(a) FROM t1} $db
78 do_test exclusive2-1.0 {
79 readPagerChangeCounter test.db
82 #-----------------------------------------------------------------------
83 # The following tests - exclusive2-1.X - check that:
85 # 1-3: Build a database with connection 1, calculate a signature.
86 # 4-9: Modify the database using a second connection in a way that
87 # does not modify the freelist, then reset the pager change-counter
88 # to the value it had before the modifications.
89 # 8: Check that using the first connection, the database signature
90 # is still the same. This is because it uses the in-memory cache.
91 # It can't tell the db has changed because we reset the change-counter.
92 # 9: Increment the change-counter.
93 # 10: Ensure that the first connection now sees the updated database. It
94 # sees the change-counter has been incremented and discards the
95 # invalid in-memory cache.
97 # This will only work if the database cache is large enough to hold
98 # the entire database. In the case of 1024 byte pages, this means
99 # the cache size must be at least 17. Otherwise, some pages will be
100 # loaded from the database file in step 8.
102 do_test exclusive2-1.1 {
105 CREATE TABLE t1(a, b);
106 INSERT INTO t1(a) VALUES(randstr(10, 400));
107 INSERT INTO t1(a) VALUES(randstr(10, 400));
108 INSERT INTO t1(a) SELECT randstr(10, 400) FROM t1;
109 INSERT INTO t1(a) SELECT randstr(10, 400) FROM t1;
110 INSERT INTO t1(a) SELECT randstr(10, 400) FROM t1;
111 INSERT INTO t1(a) SELECT randstr(10, 400) FROM t1;
112 INSERT INTO t1(a) SELECT randstr(10, 400) FROM t1;
114 SELECT count(*) FROM t1;
117 do_test exclusive2-1.2.1 {
118 # Make sure the pager cache is large enough to store the
120 set nPage [expr [file size test.db]/1024]
121 if {$::SQLITE_DEFAULT_CACHE_SIZE < $nPage} {
122 execsql "PRAGMA cache_size = $nPage"
124 expr {[execsql {PRAGMA cache_size}] >= $nPage}
126 do_test exclusive2-1.2 {
128 readPagerChangeCounter test.db
130 do_test exclusive2-1.3 {
133 do_test exclusive2-1.4 {
137 do_test exclusive2-1.5 {
139 UPDATE t1 SET b=a, a=NULL;
141 expr {[t1sig db2] eq $::sig}
143 do_test exclusive2-1.6 {
144 readPagerChangeCounter test.db
146 do_test exclusive2-1.7 {
147 pagerChangeCounter test.db 1
149 do_test exclusive2-1.9 {
151 expr {[t1sig] eq $::sig}
153 do_test exclusive2-1.10 {
154 pagerChangeCounter test.db 2
156 do_test exclusive2-1.11 {
157 expr {[t1sig] eq $::sig}
160 #--------------------------------------------------------------------
161 # These tests - exclusive2-2.X - are similar to exclusive2-1.X,
162 # except that they are run with locking_mode=EXCLUSIVE.
164 # 1-3: Build a database with exclusive-access connection 1,
165 # calculate a signature.
166 # 4: Corrupt the database by writing 10000 bytes of garbage
167 # starting at the beginning of page 2. Check that connection 1
168 # still works. It should be accessing the in-memory cache.
169 # 5-6: Modify the dataase change-counter. Connection 1 still works
170 # entirely from in-memory cache, because it doesn't check the
172 # 7-8 Set the locking-mode back to normal. After the db is unlocked,
173 # SQLite detects the modified change-counter and discards the
174 # in-memory cache. Then it finds the corruption caused in step 4....
176 # As above, this test is only applicable if the pager cache is
177 # large enough to hold the entire database. With 1024 byte pages,
178 # this means 19 pages. We also need to disable the soft-heap-limit
179 # to prevent memory-induced cache spills.
181 do_test exclusive2-2.1 {
182 execsql {PRAGMA locking_mode = exclusive;}
186 INSERT INTO t1(a) VALUES(randstr(10, 400));
187 INSERT INTO t1(a) VALUES(randstr(10, 400));
188 INSERT INTO t1(a) SELECT randstr(10, 400) FROM t1;
189 INSERT INTO t1(a) SELECT randstr(10, 400) FROM t1;
190 INSERT INTO t1(a) SELECT randstr(10, 400) FROM t1;
191 INSERT INTO t1(a) SELECT randstr(10, 400) FROM t1;
192 INSERT INTO t1(a) SELECT randstr(10, 400) FROM t1;
194 SELECT count(*) FROM t1;
197 do_test exclusive2-2.2.1 {
198 # Make sure the pager cache is large enough to store the
200 set nPage [expr [file size test.db]/1024]
201 if {$::SQLITE_DEFAULT_CACHE_SIZE < $nPage} {
202 execsql "PRAGMA cache_size = $nPage"
204 expr {[execsql {PRAGMA cache_size}] >= $nPage}
206 do_test exclusive2-2.2 {
208 readPagerChangeCounter test.db
210 do_test exclusive2-2.3 {
214 do_test exclusive2-2.4 {
215 set ::fd [open test.db RDWR]
216 fconfigure $::fd -translation binary
218 puts -nonewline $::fd [string repeat [binary format c 0] 10000]
223 do_test exclusive2-2.5 {
224 pagerChangeCounter test.db 5 $::fd
226 do_test exclusive2-2.6 {
229 do_test exclusive2-2.7 {
230 execsql {PRAGMA locking_mode = normal}
234 do_test exclusive2-2.8 {
235 set rc [catch {t1sig} msg]
237 } {1 {database disk image is malformed}}
239 #--------------------------------------------------------------------
240 # These tests - exclusive2-3.X - verify that the pager change-counter
241 # is only incremented by the first change when in exclusive access
242 # mode. In normal mode, the change-counter is incremented once
243 # per write-transaction.
249 file delete -force test.db
250 file delete -force test.db-journal
252 do_test exclusive2-3.0 {
256 CREATE TABLE t1(a UNIQUE);
257 INSERT INTO t1 VALUES(randstr(10, 400));
258 INSERT INTO t1 VALUES(randstr(10, 400));
261 readPagerChangeCounter test.db
263 do_test exclusive2-3.1 {
265 INSERT INTO t1 VALUES(randstr(10, 400));
267 readPagerChangeCounter test.db
269 do_test exclusive2-3.2 {
271 INSERT INTO t1 VALUES(randstr(10, 400));
273 readPagerChangeCounter test.db
275 do_test exclusive2-3.3 {
277 PRAGMA locking_mode = exclusive;
278 INSERT INTO t1 VALUES(randstr(10, 400));
280 readPagerChangeCounter test.db
282 do_test exclusive2-3.4 {
284 INSERT INTO t1 VALUES(randstr(10, 400));
286 readPagerChangeCounter test.db
288 do_test exclusive2-3.5 {
290 PRAGMA locking_mode = normal;
291 INSERT INTO t1 VALUES(randstr(10, 400));
293 readPagerChangeCounter test.db
295 do_test exclusive2-3.6 {
297 INSERT INTO t1 VALUES(randstr(10, 400));
299 readPagerChangeCounter test.db
301 sqlite3_soft_heap_limit $soft_limit