sl@0
|
1 |
# 2005 December 30
|
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 |
# The focus of the tests in this file are IO errors that occur in a shared
|
sl@0
|
13 |
# cache context. What happens to connection B if one connection A encounters
|
sl@0
|
14 |
# an IO-error whilst reading or writing the file-system?
|
sl@0
|
15 |
#
|
sl@0
|
16 |
# $Id: shared_err.test,v 1.23 2008/09/29 14:12:57 danielk1977 Exp $
|
sl@0
|
17 |
|
sl@0
|
18 |
proc skip {args} {}
|
sl@0
|
19 |
|
sl@0
|
20 |
|
sl@0
|
21 |
set testdir [file dirname $argv0]
|
sl@0
|
22 |
source $testdir/tester.tcl
|
sl@0
|
23 |
source $testdir/malloc_common.tcl
|
sl@0
|
24 |
db close
|
sl@0
|
25 |
|
sl@0
|
26 |
ifcapable !shared_cache||!subquery {
|
sl@0
|
27 |
finish_test
|
sl@0
|
28 |
return
|
sl@0
|
29 |
}
|
sl@0
|
30 |
|
sl@0
|
31 |
set ::enable_shared_cache [sqlite3_enable_shared_cache 1]
|
sl@0
|
32 |
|
sl@0
|
33 |
do_ioerr_test shared_ioerr-1 -tclprep {
|
sl@0
|
34 |
sqlite3 db2 test.db
|
sl@0
|
35 |
execsql {
|
sl@0
|
36 |
PRAGMA read_uncommitted = 1;
|
sl@0
|
37 |
CREATE TABLE t1(a,b,c);
|
sl@0
|
38 |
BEGIN;
|
sl@0
|
39 |
SELECT * FROM sqlite_master;
|
sl@0
|
40 |
} db2
|
sl@0
|
41 |
} -sqlbody {
|
sl@0
|
42 |
SELECT * FROM sqlite_master;
|
sl@0
|
43 |
INSERT INTO t1 VALUES(1,2,3);
|
sl@0
|
44 |
BEGIN TRANSACTION;
|
sl@0
|
45 |
INSERT INTO t1 VALUES(1,2,3);
|
sl@0
|
46 |
INSERT INTO t1 VALUES(4,5,6);
|
sl@0
|
47 |
ROLLBACK;
|
sl@0
|
48 |
SELECT * FROM t1;
|
sl@0
|
49 |
BEGIN TRANSACTION;
|
sl@0
|
50 |
INSERT INTO t1 VALUES(1,2,3);
|
sl@0
|
51 |
INSERT INTO t1 VALUES(4,5,6);
|
sl@0
|
52 |
COMMIT;
|
sl@0
|
53 |
SELECT * FROM t1;
|
sl@0
|
54 |
DELETE FROM t1 WHERE a<100;
|
sl@0
|
55 |
} -cleanup {
|
sl@0
|
56 |
do_test shared_ioerr-1.$n.cleanup.1 {
|
sl@0
|
57 |
set res [catchsql {
|
sl@0
|
58 |
SELECT * FROM t1;
|
sl@0
|
59 |
} db2]
|
sl@0
|
60 |
set possible_results [list \
|
sl@0
|
61 |
"1 {disk I/O error}" \
|
sl@0
|
62 |
"0 {1 2 3}" \
|
sl@0
|
63 |
"0 {1 2 3 1 2 3 4 5 6}" \
|
sl@0
|
64 |
"0 {1 2 3 1 2 3 4 5 6 1 2 3 4 5 6}" \
|
sl@0
|
65 |
"0 {}" \
|
sl@0
|
66 |
"1 {database disk image is malformed}" \
|
sl@0
|
67 |
]
|
sl@0
|
68 |
set rc [expr [lsearch -exact $possible_results $res] >= 0]
|
sl@0
|
69 |
if {$rc != 1} {
|
sl@0
|
70 |
puts ""
|
sl@0
|
71 |
puts "Result: $res"
|
sl@0
|
72 |
}
|
sl@0
|
73 |
set rc
|
sl@0
|
74 |
} {1}
|
sl@0
|
75 |
|
sl@0
|
76 |
# The "database disk image is malformed" is a special case that can
|
sl@0
|
77 |
# occur if an IO error occurs during a rollback in the {SELECT * FROM t1}
|
sl@0
|
78 |
# statement above. This test is to make sure there is no real database
|
sl@0
|
79 |
# corruption.
|
sl@0
|
80 |
db2 close
|
sl@0
|
81 |
do_test shared_ioerr-1.$n.cleanup.2 {
|
sl@0
|
82 |
execsql {pragma integrity_check} db
|
sl@0
|
83 |
} {ok}
|
sl@0
|
84 |
}
|
sl@0
|
85 |
|
sl@0
|
86 |
do_ioerr_test shared_ioerr-2 -tclprep {
|
sl@0
|
87 |
sqlite3 db2 test.db
|
sl@0
|
88 |
execsql {
|
sl@0
|
89 |
PRAGMA read_uncommitted = 1;
|
sl@0
|
90 |
BEGIN;
|
sl@0
|
91 |
CREATE TABLE t1(a, b);
|
sl@0
|
92 |
INSERT INTO t1(oid) VALUES(NULL);
|
sl@0
|
93 |
INSERT INTO t1(oid) SELECT NULL FROM t1;
|
sl@0
|
94 |
INSERT INTO t1(oid) SELECT NULL FROM t1;
|
sl@0
|
95 |
INSERT INTO t1(oid) SELECT NULL FROM t1;
|
sl@0
|
96 |
INSERT INTO t1(oid) SELECT NULL FROM t1;
|
sl@0
|
97 |
INSERT INTO t1(oid) SELECT NULL FROM t1;
|
sl@0
|
98 |
INSERT INTO t1(oid) SELECT NULL FROM t1;
|
sl@0
|
99 |
INSERT INTO t1(oid) SELECT NULL FROM t1;
|
sl@0
|
100 |
INSERT INTO t1(oid) SELECT NULL FROM t1;
|
sl@0
|
101 |
INSERT INTO t1(oid) SELECT NULL FROM t1;
|
sl@0
|
102 |
INSERT INTO t1(oid) SELECT NULL FROM t1;
|
sl@0
|
103 |
UPDATE t1 set a = oid, b = 'abcdefghijklmnopqrstuvwxyz0123456789';
|
sl@0
|
104 |
CREATE INDEX i1 ON t1(a);
|
sl@0
|
105 |
COMMIT;
|
sl@0
|
106 |
BEGIN;
|
sl@0
|
107 |
SELECT * FROM sqlite_master;
|
sl@0
|
108 |
} db2
|
sl@0
|
109 |
} -tclbody {
|
sl@0
|
110 |
set ::residx 0
|
sl@0
|
111 |
execsql {DELETE FROM t1 WHERE 0 = (a % 2);}
|
sl@0
|
112 |
incr ::residx
|
sl@0
|
113 |
|
sl@0
|
114 |
# When this transaction begins the table contains 512 entries. The
|
sl@0
|
115 |
# two statements together add 512+146 more if it succeeds.
|
sl@0
|
116 |
# (1024/7==146)
|
sl@0
|
117 |
execsql {BEGIN;}
|
sl@0
|
118 |
execsql {INSERT INTO t1 SELECT a+1, b FROM t1;}
|
sl@0
|
119 |
execsql {INSERT INTO t1 SELECT 'string' || a, b FROM t1 WHERE 0 = (a%7);}
|
sl@0
|
120 |
execsql {COMMIT;}
|
sl@0
|
121 |
|
sl@0
|
122 |
incr ::residx
|
sl@0
|
123 |
} -cleanup {
|
sl@0
|
124 |
catchsql ROLLBACK
|
sl@0
|
125 |
do_test shared_ioerr-2.$n.cleanup.1 {
|
sl@0
|
126 |
set res [catchsql {
|
sl@0
|
127 |
SELECT max(a), min(a), count(*) FROM (SELECT a FROM t1 order by a);
|
sl@0
|
128 |
} db2]
|
sl@0
|
129 |
set possible_results [list \
|
sl@0
|
130 |
{0 {1024 1 1024}} \
|
sl@0
|
131 |
{0 {1023 1 512}} \
|
sl@0
|
132 |
{0 {string994 1 1170}} \
|
sl@0
|
133 |
]
|
sl@0
|
134 |
set idx [lsearch -exact $possible_results $res]
|
sl@0
|
135 |
set success [expr {$idx==$::residx || $res=="1 {disk I/O error}"}]
|
sl@0
|
136 |
if {!$success} {
|
sl@0
|
137 |
puts ""
|
sl@0
|
138 |
puts "Result: \"$res\" ($::residx)"
|
sl@0
|
139 |
}
|
sl@0
|
140 |
set success
|
sl@0
|
141 |
} {1}
|
sl@0
|
142 |
db2 close
|
sl@0
|
143 |
}
|
sl@0
|
144 |
|
sl@0
|
145 |
# This test is designed to provoke an IO error when a cursor position is
|
sl@0
|
146 |
# "saved" (because another cursor is going to modify the underlying table).
|
sl@0
|
147 |
#
|
sl@0
|
148 |
do_ioerr_test shared_ioerr-3 -tclprep {
|
sl@0
|
149 |
sqlite3 db2 test.db
|
sl@0
|
150 |
execsql {
|
sl@0
|
151 |
PRAGMA read_uncommitted = 1;
|
sl@0
|
152 |
PRAGMA cache_size = 10;
|
sl@0
|
153 |
BEGIN;
|
sl@0
|
154 |
CREATE TABLE t1(a, b, UNIQUE(a, b));
|
sl@0
|
155 |
} db2
|
sl@0
|
156 |
for {set i 0} {$i < 200} {incr i} {
|
sl@0
|
157 |
set a [string range [string repeat "[format %03d $i]." 5] 0 end-1]
|
sl@0
|
158 |
|
sl@0
|
159 |
set b [string repeat $i 2000]
|
sl@0
|
160 |
execsql {INSERT INTO t1 VALUES($a, $b)} db2
|
sl@0
|
161 |
}
|
sl@0
|
162 |
execsql {COMMIT} db2
|
sl@0
|
163 |
set ::DB2 [sqlite3_connection_pointer db2]
|
sl@0
|
164 |
set ::STMT [sqlite3_prepare $::DB2 "SELECT a FROM t1 ORDER BY a" -1 DUMMY]
|
sl@0
|
165 |
sqlite3_step $::STMT ;# Cursor points at 000.000.000.000
|
sl@0
|
166 |
sqlite3_step $::STMT ;# Cursor points at 001.001.001.001
|
sl@0
|
167 |
|
sl@0
|
168 |
} -tclbody {
|
sl@0
|
169 |
execsql {
|
sl@0
|
170 |
BEGIN;
|
sl@0
|
171 |
INSERT INTO t1 VALUES('201.201.201.201.201', NULL);
|
sl@0
|
172 |
UPDATE t1 SET a = '202.202.202.202.202' WHERE a LIKE '201%';
|
sl@0
|
173 |
COMMIT;
|
sl@0
|
174 |
}
|
sl@0
|
175 |
} -cleanup {
|
sl@0
|
176 |
set ::steprc [sqlite3_step $::STMT]
|
sl@0
|
177 |
set ::column [sqlite3_column_text $::STMT 0]
|
sl@0
|
178 |
set ::finalrc [sqlite3_finalize $::STMT]
|
sl@0
|
179 |
|
sl@0
|
180 |
# There are three possible outcomes here (assuming persistent IO errors):
|
sl@0
|
181 |
#
|
sl@0
|
182 |
# 1. If the [sqlite3_step] did not require any IO (required pages in
|
sl@0
|
183 |
# the cache), then the next row ("002...") may be retrieved
|
sl@0
|
184 |
# successfully.
|
sl@0
|
185 |
#
|
sl@0
|
186 |
# 2. If the [sqlite3_step] does require IO, then [sqlite3_step] returns
|
sl@0
|
187 |
# SQLITE_ERROR and [sqlite3_finalize] returns IOERR.
|
sl@0
|
188 |
#
|
sl@0
|
189 |
# 3. If, after the initial IO error, SQLite tried to rollback the
|
sl@0
|
190 |
# active transaction and a second IO error was encountered, then
|
sl@0
|
191 |
# statement $::STMT will have been aborted. This means [sqlite3_stmt]
|
sl@0
|
192 |
# returns SQLITE_ABORT, and the statement cursor does not move. i.e.
|
sl@0
|
193 |
# [sqlite3_column] still returns the current row ("001...") and
|
sl@0
|
194 |
# [sqlite3_finalize] returns SQLITE_OK.
|
sl@0
|
195 |
#
|
sl@0
|
196 |
|
sl@0
|
197 |
do_test shared_ioerr-3.$n.cleanup.1 {
|
sl@0
|
198 |
expr {
|
sl@0
|
199 |
$::steprc eq "SQLITE_ROW" ||
|
sl@0
|
200 |
$::steprc eq "SQLITE_ERROR" ||
|
sl@0
|
201 |
$::steprc eq "SQLITE_ABORT"
|
sl@0
|
202 |
}
|
sl@0
|
203 |
} {1}
|
sl@0
|
204 |
do_test shared_ioerr-3.$n.cleanup.2 {
|
sl@0
|
205 |
expr {
|
sl@0
|
206 |
($::steprc eq "SQLITE_ROW" && $::column eq "002.002.002.002.002") ||
|
sl@0
|
207 |
($::steprc eq "SQLITE_ERROR" && $::column eq "") ||
|
sl@0
|
208 |
($::steprc eq "SQLITE_ABORT" && $::column eq "001.001.001.001.001")
|
sl@0
|
209 |
}
|
sl@0
|
210 |
} {1}
|
sl@0
|
211 |
do_test shared_ioerr-3.$n.cleanup.3 {
|
sl@0
|
212 |
expr {
|
sl@0
|
213 |
($::steprc eq "SQLITE_ROW" && $::finalrc eq "SQLITE_OK") ||
|
sl@0
|
214 |
($::steprc eq "SQLITE_ERROR" && $::finalrc eq "SQLITE_IOERR") ||
|
sl@0
|
215 |
($::steprc eq "SQLITE_ERROR" && $::finalrc eq "SQLITE_ABORT")
|
sl@0
|
216 |
}
|
sl@0
|
217 |
} {1}
|
sl@0
|
218 |
|
sl@0
|
219 |
# db2 eval {select * from sqlite_master}
|
sl@0
|
220 |
db2 close
|
sl@0
|
221 |
}
|
sl@0
|
222 |
|
sl@0
|
223 |
# This is a repeat of the previous test except that this time we
|
sl@0
|
224 |
# are doing a reverse-order scan of the table when the cursor is
|
sl@0
|
225 |
# "saved".
|
sl@0
|
226 |
#
|
sl@0
|
227 |
do_ioerr_test shared_ioerr-3rev -tclprep {
|
sl@0
|
228 |
sqlite3 db2 test.db
|
sl@0
|
229 |
execsql {
|
sl@0
|
230 |
PRAGMA read_uncommitted = 1;
|
sl@0
|
231 |
PRAGMA cache_size = 10;
|
sl@0
|
232 |
BEGIN;
|
sl@0
|
233 |
CREATE TABLE t1(a, b, UNIQUE(a, b));
|
sl@0
|
234 |
} db2
|
sl@0
|
235 |
for {set i 0} {$i < 200} {incr i} {
|
sl@0
|
236 |
set a [string range [string repeat "[format %03d $i]." 5] 0 end-1]
|
sl@0
|
237 |
|
sl@0
|
238 |
set b [string repeat $i 2000]
|
sl@0
|
239 |
execsql {INSERT INTO t1 VALUES($a, $b)} db2
|
sl@0
|
240 |
}
|
sl@0
|
241 |
execsql {COMMIT} db2
|
sl@0
|
242 |
set ::DB2 [sqlite3_connection_pointer db2]
|
sl@0
|
243 |
set ::STMT [sqlite3_prepare $::DB2 \
|
sl@0
|
244 |
"SELECT a FROM t1 ORDER BY a DESC" -1 DUMMY]
|
sl@0
|
245 |
sqlite3_step $::STMT ;# Cursor points at 199.199.199.199.199
|
sl@0
|
246 |
sqlite3_step $::STMT ;# Cursor points at 198.198.198.198.198
|
sl@0
|
247 |
|
sl@0
|
248 |
} -tclbody {
|
sl@0
|
249 |
execsql {
|
sl@0
|
250 |
BEGIN;
|
sl@0
|
251 |
INSERT INTO t1 VALUES('201.201.201.201.201', NULL);
|
sl@0
|
252 |
UPDATE t1 SET a = '202.202.202.202.202' WHERE a LIKE '201%';
|
sl@0
|
253 |
COMMIT;
|
sl@0
|
254 |
}
|
sl@0
|
255 |
} -cleanup {
|
sl@0
|
256 |
set ::steprc [sqlite3_step $::STMT]
|
sl@0
|
257 |
set ::column [sqlite3_column_text $::STMT 0]
|
sl@0
|
258 |
set ::finalrc [sqlite3_finalize $::STMT]
|
sl@0
|
259 |
|
sl@0
|
260 |
# There are three possible outcomes here (assuming persistent IO errors):
|
sl@0
|
261 |
#
|
sl@0
|
262 |
# 1. If the [sqlite3_step] did not require any IO (required pages in
|
sl@0
|
263 |
# the cache), then the next row ("002...") may be retrieved
|
sl@0
|
264 |
# successfully.
|
sl@0
|
265 |
#
|
sl@0
|
266 |
# 2. If the [sqlite3_step] does require IO, then [sqlite3_step] returns
|
sl@0
|
267 |
# SQLITE_ERROR and [sqlite3_finalize] returns IOERR.
|
sl@0
|
268 |
#
|
sl@0
|
269 |
# 3. If, after the initial IO error, SQLite tried to rollback the
|
sl@0
|
270 |
# active transaction and a second IO error was encountered, then
|
sl@0
|
271 |
# statement $::STMT will have been aborted. This means [sqlite3_stmt]
|
sl@0
|
272 |
# returns SQLITE_ABORT, and the statement cursor does not move. i.e.
|
sl@0
|
273 |
# [sqlite3_column] still returns the current row ("001...") and
|
sl@0
|
274 |
# [sqlite3_finalize] returns SQLITE_OK.
|
sl@0
|
275 |
#
|
sl@0
|
276 |
|
sl@0
|
277 |
do_test shared_ioerr-3rev.$n.cleanup.1 {
|
sl@0
|
278 |
expr {
|
sl@0
|
279 |
$::steprc eq "SQLITE_ROW" ||
|
sl@0
|
280 |
$::steprc eq "SQLITE_ERROR" ||
|
sl@0
|
281 |
$::steprc eq "SQLITE_ABORT"
|
sl@0
|
282 |
}
|
sl@0
|
283 |
} {1}
|
sl@0
|
284 |
do_test shared_ioerr-3rev.$n.cleanup.2 {
|
sl@0
|
285 |
expr {
|
sl@0
|
286 |
($::steprc eq "SQLITE_ROW" && $::column eq "197.197.197.197.197") ||
|
sl@0
|
287 |
($::steprc eq "SQLITE_ERROR" && $::column eq "") ||
|
sl@0
|
288 |
($::steprc eq "SQLITE_ABORT" && $::column eq "198.198.198.198.198")
|
sl@0
|
289 |
}
|
sl@0
|
290 |
} {1}
|
sl@0
|
291 |
do_test shared_ioerr-3rev.$n.cleanup.3 {
|
sl@0
|
292 |
expr {
|
sl@0
|
293 |
($::steprc eq "SQLITE_ROW" && $::finalrc eq "SQLITE_OK") ||
|
sl@0
|
294 |
($::steprc eq "SQLITE_ERROR" && $::finalrc eq "SQLITE_IOERR") ||
|
sl@0
|
295 |
($::steprc eq "SQLITE_ERROR" && $::finalrc eq "SQLITE_ABORT")
|
sl@0
|
296 |
}
|
sl@0
|
297 |
} {1}
|
sl@0
|
298 |
|
sl@0
|
299 |
# db2 eval {select * from sqlite_master}
|
sl@0
|
300 |
db2 close
|
sl@0
|
301 |
}
|
sl@0
|
302 |
|
sl@0
|
303 |
# Provoke a malloc() failure when a cursor position is being saved. This
|
sl@0
|
304 |
# only happens with index cursors (because they malloc() space to save the
|
sl@0
|
305 |
# current key value). It does not happen with tables, because an integer
|
sl@0
|
306 |
# key does not require a malloc() to store.
|
sl@0
|
307 |
#
|
sl@0
|
308 |
# The library should return an SQLITE_NOMEM to the caller. The query that
|
sl@0
|
309 |
# owns the cursor (the one for which the position is not saved) should
|
sl@0
|
310 |
# continue unaffected.
|
sl@0
|
311 |
#
|
sl@0
|
312 |
do_malloc_test shared_err-4 -tclprep {
|
sl@0
|
313 |
sqlite3 db2 test.db
|
sl@0
|
314 |
execsql {
|
sl@0
|
315 |
PRAGMA read_uncommitted = 1;
|
sl@0
|
316 |
BEGIN;
|
sl@0
|
317 |
CREATE TABLE t1(a, b, UNIQUE(a, b));
|
sl@0
|
318 |
} db2
|
sl@0
|
319 |
for {set i 0} {$i < 5} {incr i} {
|
sl@0
|
320 |
set a [string repeat $i 10]
|
sl@0
|
321 |
set b [string repeat $i 2000]
|
sl@0
|
322 |
execsql {INSERT INTO t1 VALUES($a, $b)} db2
|
sl@0
|
323 |
}
|
sl@0
|
324 |
execsql {COMMIT} db2
|
sl@0
|
325 |
set ::DB2 [sqlite3_connection_pointer db2]
|
sl@0
|
326 |
set ::STMT [sqlite3_prepare $::DB2 "SELECT a FROM t1 ORDER BY a" -1 DUMMY]
|
sl@0
|
327 |
sqlite3_step $::STMT ;# Cursor points at 0000000000
|
sl@0
|
328 |
sqlite3_step $::STMT ;# Cursor points at 1111111111
|
sl@0
|
329 |
} -tclbody {
|
sl@0
|
330 |
execsql {
|
sl@0
|
331 |
INSERT INTO t1 VALUES(6, NULL);
|
sl@0
|
332 |
}
|
sl@0
|
333 |
} -cleanup {
|
sl@0
|
334 |
do_test shared_malloc-4.$::n.cleanup.1 {
|
sl@0
|
335 |
set ::rc [sqlite3_step $::STMT]
|
sl@0
|
336 |
expr {$::rc=="SQLITE_ROW" || $::rc=="SQLITE_ERROR"}
|
sl@0
|
337 |
} {1}
|
sl@0
|
338 |
if {$::rc=="SQLITE_ROW"} {
|
sl@0
|
339 |
do_test shared_malloc-4.$::n.cleanup.2 {
|
sl@0
|
340 |
sqlite3_column_text $::STMT 0
|
sl@0
|
341 |
} {2222222222}
|
sl@0
|
342 |
}
|
sl@0
|
343 |
do_test shared_malloc-4.$::n.cleanup.3 {
|
sl@0
|
344 |
set rc [sqlite3_finalize $::STMT]
|
sl@0
|
345 |
expr {$rc=="SQLITE_OK" || $rc=="SQLITE_ABORT" ||
|
sl@0
|
346 |
$rc=="SQLITE_NOMEM" || $rc=="SQLITE_IOERR"}
|
sl@0
|
347 |
} {1}
|
sl@0
|
348 |
# db2 eval {select * from sqlite_master}
|
sl@0
|
349 |
db2 close
|
sl@0
|
350 |
}
|
sl@0
|
351 |
|
sl@0
|
352 |
do_malloc_test shared_err-5 -tclbody {
|
sl@0
|
353 |
db close
|
sl@0
|
354 |
sqlite3 dbX test.db
|
sl@0
|
355 |
sqlite3 dbY test.db
|
sl@0
|
356 |
dbX close
|
sl@0
|
357 |
dbY close
|
sl@0
|
358 |
} -cleanup {
|
sl@0
|
359 |
catch {dbX close}
|
sl@0
|
360 |
catch {dbY close}
|
sl@0
|
361 |
}
|
sl@0
|
362 |
|
sl@0
|
363 |
do_malloc_test shared_err-6 -tclbody {
|
sl@0
|
364 |
catch {db close}
|
sl@0
|
365 |
sqlite3_thread_cleanup
|
sl@0
|
366 |
sqlite3_enable_shared_cache 0
|
sl@0
|
367 |
} -cleanup {
|
sl@0
|
368 |
sqlite3_enable_shared_cache 1
|
sl@0
|
369 |
}
|
sl@0
|
370 |
|
sl@0
|
371 |
# As of 3.5.0, sqlite3_enable_shared_cache can be called at
|
sl@0
|
372 |
# any time and from any thread
|
sl@0
|
373 |
#do_test shared_err-misuse-7.1 {
|
sl@0
|
374 |
# sqlite3 db test.db
|
sl@0
|
375 |
# catch {
|
sl@0
|
376 |
# sqlite3_enable_shared_cache 0
|
sl@0
|
377 |
# } msg
|
sl@0
|
378 |
# set msg
|
sl@0
|
379 |
#} {library routine called out of sequence}
|
sl@0
|
380 |
|
sl@0
|
381 |
# Again provoke a malloc() failure when a cursor position is being saved,
|
sl@0
|
382 |
# this time during a ROLLBACK operation by some other handle.
|
sl@0
|
383 |
#
|
sl@0
|
384 |
# The library should return an SQLITE_NOMEM to the caller. The query that
|
sl@0
|
385 |
# owns the cursor (the one for which the position is not saved) should
|
sl@0
|
386 |
# be aborted.
|
sl@0
|
387 |
#
|
sl@0
|
388 |
set ::aborted 0
|
sl@0
|
389 |
do_malloc_test shared_err-8 -tclprep {
|
sl@0
|
390 |
sqlite3 db2 test.db
|
sl@0
|
391 |
execsql {
|
sl@0
|
392 |
PRAGMA read_uncommitted = 1;
|
sl@0
|
393 |
BEGIN;
|
sl@0
|
394 |
CREATE TABLE t1(a, b, UNIQUE(a, b));
|
sl@0
|
395 |
} db2
|
sl@0
|
396 |
for {set i 0} {$i < 2} {incr i} {
|
sl@0
|
397 |
set a [string repeat $i 10]
|
sl@0
|
398 |
set b [string repeat $i 2000]
|
sl@0
|
399 |
execsql {INSERT INTO t1 VALUES($a, $b)} db2
|
sl@0
|
400 |
}
|
sl@0
|
401 |
execsql {COMMIT} db2
|
sl@0
|
402 |
set ::DB2 [sqlite3_connection_pointer db2]
|
sl@0
|
403 |
set ::STMT [sqlite3_prepare $::DB2 "SELECT a FROM t1 ORDER BY a" -1 DUMMY]
|
sl@0
|
404 |
sqlite3_step $::STMT ;# Cursor points at 0000000000
|
sl@0
|
405 |
sqlite3_step $::STMT ;# Cursor points at 1111111111
|
sl@0
|
406 |
} -tclbody {
|
sl@0
|
407 |
execsql {
|
sl@0
|
408 |
BEGIN;
|
sl@0
|
409 |
INSERT INTO t1 VALUES(6, NULL);
|
sl@0
|
410 |
ROLLBACK;
|
sl@0
|
411 |
}
|
sl@0
|
412 |
} -cleanup {
|
sl@0
|
413 |
# UPDATE: As of [5668], if the rollback fails SQLITE_CORRUPT is returned.
|
sl@0
|
414 |
# So these tests have been updated to expect SQLITE_CORRUPT and its
|
sl@0
|
415 |
# associated English language error message.
|
sl@0
|
416 |
#
|
sl@0
|
417 |
do_test shared_malloc-8.$::n.cleanup.1 {
|
sl@0
|
418 |
set res [catchsql {SELECT a FROM t1} db2]
|
sl@0
|
419 |
set ans [lindex $res 1]
|
sl@0
|
420 |
if {[lindex $res 0]} {
|
sl@0
|
421 |
set r [expr {
|
sl@0
|
422 |
$ans=="disk I/O error" ||
|
sl@0
|
423 |
$ans=="out of memory" ||
|
sl@0
|
424 |
$ans=="database disk image is malformed"
|
sl@0
|
425 |
}]
|
sl@0
|
426 |
} else {
|
sl@0
|
427 |
set r [expr {[lrange $ans 0 1]=="0000000000 1111111111"}]
|
sl@0
|
428 |
}
|
sl@0
|
429 |
} {1}
|
sl@0
|
430 |
do_test shared_malloc-8.$::n.cleanup.2 {
|
sl@0
|
431 |
set rc1 [sqlite3_step $::STMT]
|
sl@0
|
432 |
set rc2 [sqlite3_finalize $::STMT]
|
sl@0
|
433 |
if {$rc2=="SQLITE_ABORT"} {
|
sl@0
|
434 |
incr ::aborted
|
sl@0
|
435 |
}
|
sl@0
|
436 |
expr {
|
sl@0
|
437 |
($rc1=="SQLITE_DONE" && $rc2=="SQLITE_OK") ||
|
sl@0
|
438 |
($rc1=="SQLITE_ERROR" && $rc2=="SQLITE_ABORT") ||
|
sl@0
|
439 |
($rc1=="SQLITE_ERROR" && $rc2=="SQLITE_NOMEM") ||
|
sl@0
|
440 |
($rc1=="SQLITE_ERROR" && $rc2=="SQLITE_IOERR") ||
|
sl@0
|
441 |
($rc1=="SQLITE_ERROR" && $rc2=="SQLITE_CORRUPT")
|
sl@0
|
442 |
}
|
sl@0
|
443 |
} {1}
|
sl@0
|
444 |
db2 close
|
sl@0
|
445 |
}
|
sl@0
|
446 |
do_test shared_malloc-8.X {
|
sl@0
|
447 |
# Test that one or more queries were aborted due to the malloc() failure.
|
sl@0
|
448 |
expr $::aborted>=1
|
sl@0
|
449 |
} {1}
|
sl@0
|
450 |
|
sl@0
|
451 |
# This test is designed to catch a specific bug that was present during
|
sl@0
|
452 |
# development of 3.5.0. If a malloc() failed while setting the page-size,
|
sl@0
|
453 |
# a buffer (Pager.pTmpSpace) was being freed. This could cause a seg-fault
|
sl@0
|
454 |
# later if another connection tried to use the pager.
|
sl@0
|
455 |
#
|
sl@0
|
456 |
# This test will crash 3.4.2.
|
sl@0
|
457 |
#
|
sl@0
|
458 |
do_malloc_test shared_err-9 -tclprep {
|
sl@0
|
459 |
sqlite3 db2 test.db
|
sl@0
|
460 |
} -sqlbody {
|
sl@0
|
461 |
PRAGMA page_size = 4096;
|
sl@0
|
462 |
PRAGMA page_size = 1024;
|
sl@0
|
463 |
} -cleanup {
|
sl@0
|
464 |
db2 eval {
|
sl@0
|
465 |
CREATE TABLE abc(a, b, c);
|
sl@0
|
466 |
BEGIN;
|
sl@0
|
467 |
INSERT INTO abc VALUES(1, 2, 3);
|
sl@0
|
468 |
ROLLBACK;
|
sl@0
|
469 |
}
|
sl@0
|
470 |
db2 close
|
sl@0
|
471 |
}
|
sl@0
|
472 |
|
sl@0
|
473 |
catch {db close}
|
sl@0
|
474 |
catch {db2 close}
|
sl@0
|
475 |
do_malloc_test shared_err-10 -tclprep {
|
sl@0
|
476 |
sqlite3 db test.db
|
sl@0
|
477 |
sqlite3 db2 test.db
|
sl@0
|
478 |
|
sl@0
|
479 |
db eval { SELECT * FROM sqlite_master }
|
sl@0
|
480 |
db2 eval {
|
sl@0
|
481 |
BEGIN;
|
sl@0
|
482 |
CREATE TABLE abc(a, b, c);
|
sl@0
|
483 |
}
|
sl@0
|
484 |
} -tclbody {
|
sl@0
|
485 |
catch {db eval {SELECT * FROM sqlite_master}}
|
sl@0
|
486 |
error 1
|
sl@0
|
487 |
} -cleanup {
|
sl@0
|
488 |
execsql { SELECT * FROM sqlite_master }
|
sl@0
|
489 |
}
|
sl@0
|
490 |
|
sl@0
|
491 |
do_malloc_test shared_err-11 -tclprep {
|
sl@0
|
492 |
sqlite3 db test.db
|
sl@0
|
493 |
sqlite3 db2 test.db
|
sl@0
|
494 |
|
sl@0
|
495 |
db eval { SELECT * FROM sqlite_master }
|
sl@0
|
496 |
db2 eval {
|
sl@0
|
497 |
BEGIN;
|
sl@0
|
498 |
CREATE TABLE abc(a, b, c);
|
sl@0
|
499 |
}
|
sl@0
|
500 |
} -tclbody {
|
sl@0
|
501 |
catch {db eval {SELECT * FROM sqlite_master}}
|
sl@0
|
502 |
catch {sqlite3_errmsg16 db}
|
sl@0
|
503 |
error 1
|
sl@0
|
504 |
} -cleanup {
|
sl@0
|
505 |
execsql { SELECT * FROM sqlite_master }
|
sl@0
|
506 |
}
|
sl@0
|
507 |
|
sl@0
|
508 |
catch {db close}
|
sl@0
|
509 |
catch {db2 close}
|
sl@0
|
510 |
|
sl@0
|
511 |
do_malloc_test shared_err-12 -sqlbody {
|
sl@0
|
512 |
CREATE TABLE abc(a, b, c);
|
sl@0
|
513 |
INSERT INTO abc VALUES(1, 2, 3);
|
sl@0
|
514 |
}
|
sl@0
|
515 |
|
sl@0
|
516 |
catch {db close}
|
sl@0
|
517 |
catch {db2 close}
|
sl@0
|
518 |
sqlite3_enable_shared_cache $::enable_shared_cache
|
sl@0
|
519 |
finish_test
|