sl@0
|
1 |
# 2007 August 23
|
sl@0
|
2 |
#
|
sl@0
|
3 |
# The author disclaims copyright to this source code. In place of
|
sl@0
|
4 |
# a legal notice, here is a blessing:
|
sl@0
|
5 |
#
|
sl@0
|
6 |
# May you do good and not evil.
|
sl@0
|
7 |
# May you find forgiveness for yourself and forgive others.
|
sl@0
|
8 |
# May you share freely, never taking more than you give.
|
sl@0
|
9 |
#
|
sl@0
|
10 |
#***********************************************************************
|
sl@0
|
11 |
#
|
sl@0
|
12 |
# This file contains tests that verify that SQLite can correctly rollback
|
sl@0
|
13 |
# databases after crashes when using the special IO modes triggered
|
sl@0
|
14 |
# by device IOCAP flags.
|
sl@0
|
15 |
#
|
sl@0
|
16 |
# $Id: crash3.test,v 1.4 2008/07/12 14:52:20 drh Exp $
|
sl@0
|
17 |
|
sl@0
|
18 |
set testdir [file dirname $argv0]
|
sl@0
|
19 |
source $testdir/tester.tcl
|
sl@0
|
20 |
|
sl@0
|
21 |
ifcapable !crashtest {
|
sl@0
|
22 |
finish_test
|
sl@0
|
23 |
return
|
sl@0
|
24 |
}
|
sl@0
|
25 |
|
sl@0
|
26 |
proc do_test2 {name tcl res1 res2} {
|
sl@0
|
27 |
set script [subst -nocommands {
|
sl@0
|
28 |
do_test $name {
|
sl@0
|
29 |
set res1 {$res1}
|
sl@0
|
30 |
set res2 {$res2}
|
sl@0
|
31 |
set res [eval {$tcl}]
|
sl@0
|
32 |
if {[set res] eq [set res1] || [set res] eq [set res2]} {
|
sl@0
|
33 |
set res "{[set res1]} or {[set res2]}"
|
sl@0
|
34 |
}
|
sl@0
|
35 |
set res
|
sl@0
|
36 |
} {{$res1} or {$res2}}
|
sl@0
|
37 |
}]
|
sl@0
|
38 |
uplevel $script
|
sl@0
|
39 |
}
|
sl@0
|
40 |
|
sl@0
|
41 |
# This block tests crash-recovery when the IOCAP_ATOMIC flags is set.
|
sl@0
|
42 |
#
|
sl@0
|
43 |
# Each iteration of the following loop sets up the database to contain
|
sl@0
|
44 |
# the following schema and data:
|
sl@0
|
45 |
#
|
sl@0
|
46 |
# CREATE TABLE abc(a, b, c);
|
sl@0
|
47 |
# INSERT INTO abc VALUES(1, 2, 3);
|
sl@0
|
48 |
#
|
sl@0
|
49 |
# Then execute the SQL statement, scheduling a crash for part-way through
|
sl@0
|
50 |
# the first sync() of either the database file or the journal file (often
|
sl@0
|
51 |
# the journal file is not required - meaning no crash occurs).
|
sl@0
|
52 |
#
|
sl@0
|
53 |
# After the crash (or absence of a crash), open the database and
|
sl@0
|
54 |
# verify that:
|
sl@0
|
55 |
#
|
sl@0
|
56 |
# * The integrity check passes, and
|
sl@0
|
57 |
# * The contents of table abc is either {1 2 3} or the value specified
|
sl@0
|
58 |
# to the right of the SQL statement below.
|
sl@0
|
59 |
#
|
sl@0
|
60 |
# The procedure is repeated 10 times for each SQL statement. Five times
|
sl@0
|
61 |
# with the crash scheduled for midway through the first journal sync (if
|
sl@0
|
62 |
# any), and five times with the crash midway through the database sync.
|
sl@0
|
63 |
#
|
sl@0
|
64 |
set tn 1
|
sl@0
|
65 |
foreach {sql res2} [list \
|
sl@0
|
66 |
{INSERT INTO abc VALUES(4, 5, 6)} {1 2 3 4 5 6} \
|
sl@0
|
67 |
{DELETE FROM abc} {} \
|
sl@0
|
68 |
{INSERT INTO abc SELECT * FROM abc} {1 2 3 1 2 3} \
|
sl@0
|
69 |
{UPDATE abc SET a = 2} {2 2 3} \
|
sl@0
|
70 |
{INSERT INTO abc VALUES(4, 5, randstr(1000,1000))} {n/a} \
|
sl@0
|
71 |
{CREATE TABLE def(d, e, f)} {n/a} \
|
sl@0
|
72 |
] {
|
sl@0
|
73 |
for {set ii 0} {$ii < 10} {incr ii} {
|
sl@0
|
74 |
|
sl@0
|
75 |
db close
|
sl@0
|
76 |
file delete -force test.db test.db-journal
|
sl@0
|
77 |
sqlite3 db test.db
|
sl@0
|
78 |
do_test crash3-1.$tn.1 {
|
sl@0
|
79 |
execsql {
|
sl@0
|
80 |
PRAGMA page_size = 1024;
|
sl@0
|
81 |
BEGIN;
|
sl@0
|
82 |
CREATE TABLE abc(a, b, c);
|
sl@0
|
83 |
INSERT INTO abc VALUES(1, 2, 3);
|
sl@0
|
84 |
COMMIT;
|
sl@0
|
85 |
}
|
sl@0
|
86 |
} {}
|
sl@0
|
87 |
db close
|
sl@0
|
88 |
|
sl@0
|
89 |
set crashfile test.db
|
sl@0
|
90 |
if {($ii%2)==0} { append crashfile -journal }
|
sl@0
|
91 |
set rand "SELECT randstr($tn,$tn);"
|
sl@0
|
92 |
do_test crash3-1.$tn.2 [subst {
|
sl@0
|
93 |
crashsql -file $crashfile -char atomic {$rand $sql}
|
sl@0
|
94 |
sqlite3 db test.db
|
sl@0
|
95 |
execsql { PRAGMA integrity_check; }
|
sl@0
|
96 |
}] {ok}
|
sl@0
|
97 |
|
sl@0
|
98 |
do_test2 crash3-1.$tn.3 {
|
sl@0
|
99 |
execsql { SELECT * FROM abc }
|
sl@0
|
100 |
} {1 2 3} $res2
|
sl@0
|
101 |
|
sl@0
|
102 |
incr tn
|
sl@0
|
103 |
}
|
sl@0
|
104 |
}
|
sl@0
|
105 |
|
sl@0
|
106 |
# This block tests both the IOCAP_SEQUENTIAL and IOCAP_SAFE_APPEND flags.
|
sl@0
|
107 |
#
|
sl@0
|
108 |
db close
|
sl@0
|
109 |
file delete -force test.db test.db-journal
|
sl@0
|
110 |
sqlite3 db test.db
|
sl@0
|
111 |
do_test crash3-2.0 {
|
sl@0
|
112 |
execsql {
|
sl@0
|
113 |
BEGIN;
|
sl@0
|
114 |
CREATE TABLE abc(a PRIMARY KEY, b, c);
|
sl@0
|
115 |
CREATE TABLE def(d PRIMARY KEY, e, f);
|
sl@0
|
116 |
PRAGMA default_cache_size = 10;
|
sl@0
|
117 |
INSERT INTO abc VALUES(randstr(10,1000),randstr(10,1000),randstr(10,1000));
|
sl@0
|
118 |
INSERT INTO abc
|
sl@0
|
119 |
SELECT randstr(10,1000),randstr(10,1000),randstr(10,1000) FROM abc;
|
sl@0
|
120 |
INSERT INTO abc
|
sl@0
|
121 |
SELECT randstr(10,1000),randstr(10,1000),randstr(10,1000) FROM abc;
|
sl@0
|
122 |
INSERT INTO abc
|
sl@0
|
123 |
SELECT randstr(10,1000),randstr(10,1000),randstr(10,1000) FROM abc;
|
sl@0
|
124 |
INSERT INTO abc
|
sl@0
|
125 |
SELECT randstr(10,1000),randstr(10,1000),randstr(10,1000) FROM abc;
|
sl@0
|
126 |
INSERT INTO abc
|
sl@0
|
127 |
SELECT randstr(10,1000),randstr(10,1000),randstr(10,1000) FROM abc;
|
sl@0
|
128 |
INSERT INTO abc
|
sl@0
|
129 |
SELECT randstr(10,1000),randstr(10,1000),randstr(10,1000) FROM abc;
|
sl@0
|
130 |
COMMIT;
|
sl@0
|
131 |
}
|
sl@0
|
132 |
} {}
|
sl@0
|
133 |
|
sl@0
|
134 |
set tn 1
|
sl@0
|
135 |
foreach {::crashfile ::delay ::char} {
|
sl@0
|
136 |
test.db 1 sequential
|
sl@0
|
137 |
test.db 1 safe_append
|
sl@0
|
138 |
test.db-journal 1 sequential
|
sl@0
|
139 |
test.db-journal 1 safe_append
|
sl@0
|
140 |
test.db-journal 2 safe_append
|
sl@0
|
141 |
test.db-journal 2 sequential
|
sl@0
|
142 |
test.db-journal 3 sequential
|
sl@0
|
143 |
test.db-journal 3 safe_append
|
sl@0
|
144 |
} {
|
sl@0
|
145 |
for {set ii 0} {$ii < 100} {incr ii} {
|
sl@0
|
146 |
set ::SQL [subst {
|
sl@0
|
147 |
SELECT randstr($ii,$ii+10);
|
sl@0
|
148 |
BEGIN;
|
sl@0
|
149 |
DELETE FROM abc WHERE random()%5;
|
sl@0
|
150 |
INSERT INTO abc
|
sl@0
|
151 |
SELECT randstr(10,1000),randstr(10,1000),randstr(10,1000)
|
sl@0
|
152 |
FROM abc
|
sl@0
|
153 |
WHERE (random()%5)==0;
|
sl@0
|
154 |
DELETE FROM def WHERE random()%5;
|
sl@0
|
155 |
INSERT INTO def
|
sl@0
|
156 |
SELECT randstr(10,1000),randstr(10,1000),randstr(10,1000)
|
sl@0
|
157 |
FROM def
|
sl@0
|
158 |
WHERE (random()%5)==0;
|
sl@0
|
159 |
COMMIT;
|
sl@0
|
160 |
}]
|
sl@0
|
161 |
|
sl@0
|
162 |
do_test crash3-2.$tn.$ii {
|
sl@0
|
163 |
crashsql -file $::crashfile -delay $::delay -char $::char $::SQL
|
sl@0
|
164 |
db close
|
sl@0
|
165 |
sqlite3 db test.db
|
sl@0
|
166 |
execsql {PRAGMA integrity_check}
|
sl@0
|
167 |
} {ok}
|
sl@0
|
168 |
}
|
sl@0
|
169 |
incr tn
|
sl@0
|
170 |
}
|
sl@0
|
171 |
|
sl@0
|
172 |
# The following block tests an interaction between IOCAP_ATOMIC and
|
sl@0
|
173 |
# IOCAP_SEQUENTIAL. At one point, if both flags were set, small
|
sl@0
|
174 |
# journal files that contained only a single page, but were required
|
sl@0
|
175 |
# for some other reason (i.e. nTrunk) were not being written to
|
sl@0
|
176 |
# disk.
|
sl@0
|
177 |
#
|
sl@0
|
178 |
for {set ii 0} {$ii < 10} {incr ii} {
|
sl@0
|
179 |
db close
|
sl@0
|
180 |
file delete -force test.db test.db-journal
|
sl@0
|
181 |
crashsql -file test.db -char {sequential atomic} {
|
sl@0
|
182 |
CREATE TABLE abc(a, b, c);
|
sl@0
|
183 |
}
|
sl@0
|
184 |
sqlite3 db test.db
|
sl@0
|
185 |
do_test crash3-3.$ii {
|
sl@0
|
186 |
execsql {PRAGMA integrity_check}
|
sl@0
|
187 |
} {ok}
|
sl@0
|
188 |
}
|
sl@0
|
189 |
|
sl@0
|
190 |
finish_test
|