sl@0
|
1 |
# 2005 November 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 |
# This file contains tests to ensure that the library handles malloc() failures
|
sl@0
|
13 |
# correctly. The emphasis of these tests are the _prepare(), _step() and
|
sl@0
|
14 |
# _finalize() calls.
|
sl@0
|
15 |
#
|
sl@0
|
16 |
# $Id: malloc3.test,v 1.23 2008/05/13 19:41:54 shane Exp $
|
sl@0
|
17 |
|
sl@0
|
18 |
set testdir [file dirname $argv0]
|
sl@0
|
19 |
source $testdir/tester.tcl
|
sl@0
|
20 |
source $testdir/malloc_common.tcl
|
sl@0
|
21 |
|
sl@0
|
22 |
# Only run these tests if memory debugging is turned on.
|
sl@0
|
23 |
#
|
sl@0
|
24 |
if {!$MEMDEBUG} {
|
sl@0
|
25 |
puts "Skipping malloc3 tests: not compiled with -DSQLITE_MEMDEBUG..."
|
sl@0
|
26 |
finish_test
|
sl@0
|
27 |
return
|
sl@0
|
28 |
}
|
sl@0
|
29 |
|
sl@0
|
30 |
#--------------------------------------------------------------------------
|
sl@0
|
31 |
# NOTES ON RECOVERING FROM A MALLOC FAILURE
|
sl@0
|
32 |
#
|
sl@0
|
33 |
# The tests in this file test the behaviours described in the following
|
sl@0
|
34 |
# paragraphs. These tests test the behaviour of the system when malloc() fails
|
sl@0
|
35 |
# inside of a call to _prepare(), _step(), _finalize() or _reset(). The
|
sl@0
|
36 |
# handling of malloc() failures within ancillary procedures is tested
|
sl@0
|
37 |
# elsewhere.
|
sl@0
|
38 |
#
|
sl@0
|
39 |
# Overview:
|
sl@0
|
40 |
#
|
sl@0
|
41 |
# Executing a statement is done in three stages (prepare, step and finalize). A
|
sl@0
|
42 |
# malloc() failure may occur within any stage. If a memory allocation fails
|
sl@0
|
43 |
# during statement preparation, no statement handle is returned. From the users
|
sl@0
|
44 |
# point of view the system state is as if _prepare() had never been called.
|
sl@0
|
45 |
#
|
sl@0
|
46 |
# If the memory allocation fails during the _step() or _finalize() calls, then
|
sl@0
|
47 |
# the database may be left in one of two states (after finalize() has been
|
sl@0
|
48 |
# called):
|
sl@0
|
49 |
#
|
sl@0
|
50 |
# * As if the neither _step() nor _finalize() had ever been called on
|
sl@0
|
51 |
# the statement handle (i.e. any changes made by the statement are
|
sl@0
|
52 |
# rolled back).
|
sl@0
|
53 |
# * The current transaction may be rolled back. In this case a hot-journal
|
sl@0
|
54 |
# may or may not actually be present in the filesystem.
|
sl@0
|
55 |
#
|
sl@0
|
56 |
# The caller can tell the difference between these two scenarios by invoking
|
sl@0
|
57 |
# _get_autocommit().
|
sl@0
|
58 |
#
|
sl@0
|
59 |
#
|
sl@0
|
60 |
# Handling of sqlite3_reset():
|
sl@0
|
61 |
#
|
sl@0
|
62 |
# If a malloc() fails while executing an sqlite3_reset() call, this is handled
|
sl@0
|
63 |
# in the same way as a failure within _finalize(). The statement handle
|
sl@0
|
64 |
# is not deleted and must be passed to _finalize() for resource deallocation.
|
sl@0
|
65 |
# Attempting to _step() or _reset() the statement after a failed _reset() will
|
sl@0
|
66 |
# always return SQLITE_NOMEM.
|
sl@0
|
67 |
#
|
sl@0
|
68 |
#
|
sl@0
|
69 |
# Other active SQL statements:
|
sl@0
|
70 |
#
|
sl@0
|
71 |
# The effect of a malloc failure on concurrently executing SQL statements,
|
sl@0
|
72 |
# particularly when the statement is executing with READ_UNCOMMITTED set and
|
sl@0
|
73 |
# the malloc() failure mandates statement rollback only. Currently, if
|
sl@0
|
74 |
# transaction rollback is required, all other vdbe's are aborted.
|
sl@0
|
75 |
#
|
sl@0
|
76 |
# Non-transient mallocs in btree.c:
|
sl@0
|
77 |
# * The Btree structure itself
|
sl@0
|
78 |
# * Each BtCursor structure
|
sl@0
|
79 |
#
|
sl@0
|
80 |
# Mallocs in pager.c:
|
sl@0
|
81 |
# readMasterJournal() - Space to read the master journal name
|
sl@0
|
82 |
# pager_delmaster() - Space for the entire master journal file
|
sl@0
|
83 |
#
|
sl@0
|
84 |
# sqlite3pager_open() - The pager structure itself
|
sl@0
|
85 |
# sqlite3_pagerget() - Space for a new page
|
sl@0
|
86 |
# pager_open_journal() - Pager.aInJournal[] bitmap
|
sl@0
|
87 |
# sqlite3pager_write() - For in-memory databases only: history page and
|
sl@0
|
88 |
# statement history page.
|
sl@0
|
89 |
# pager_stmt_begin() - Pager.aInStmt[] bitmap
|
sl@0
|
90 |
#
|
sl@0
|
91 |
# None of the above are a huge problem. The most troublesome failures are the
|
sl@0
|
92 |
# transient malloc() calls in btree.c, which can occur during the tree-balance
|
sl@0
|
93 |
# operation. This means the tree being balanced will be internally inconsistent
|
sl@0
|
94 |
# after the malloc() fails. To avoid the corrupt tree being read by a
|
sl@0
|
95 |
# READ_UNCOMMITTED query, we have to make sure the transaction or statement
|
sl@0
|
96 |
# rollback occurs before sqlite3_step() returns, not during a subsequent
|
sl@0
|
97 |
# sqlite3_finalize().
|
sl@0
|
98 |
#--------------------------------------------------------------------------
|
sl@0
|
99 |
|
sl@0
|
100 |
#--------------------------------------------------------------------------
|
sl@0
|
101 |
# NOTES ON TEST IMPLEMENTATION
|
sl@0
|
102 |
#
|
sl@0
|
103 |
# The tests in this file are implemented differently from those in other
|
sl@0
|
104 |
# files. Instead, tests are specified using three primitives: SQL, PREP and
|
sl@0
|
105 |
# TEST. Each primitive has a single argument. Primitives are processed in
|
sl@0
|
106 |
# the order they are specified in the file.
|
sl@0
|
107 |
#
|
sl@0
|
108 |
# A TEST primitive specifies a TCL script as its argument. When a TEST
|
sl@0
|
109 |
# directive is encountered the Tcl script is evaluated. Usually, this Tcl
|
sl@0
|
110 |
# script contains one or more calls to [do_test].
|
sl@0
|
111 |
#
|
sl@0
|
112 |
# A PREP primitive specifies an SQL script as its argument. When a PREP
|
sl@0
|
113 |
# directive is encountered the SQL is evaluated using database connection
|
sl@0
|
114 |
# [db].
|
sl@0
|
115 |
#
|
sl@0
|
116 |
# The SQL primitives are where the action happens. An SQL primitive must
|
sl@0
|
117 |
# contain a single, valid SQL statement as its argument. When an SQL
|
sl@0
|
118 |
# primitive is encountered, it is evaluated one or more times to test the
|
sl@0
|
119 |
# behaviour of the system when malloc() fails during preparation or
|
sl@0
|
120 |
# execution of said statement. The Nth time the statement is executed,
|
sl@0
|
121 |
# the Nth malloc is said to fail. The statement is executed until it
|
sl@0
|
122 |
# succeeds, i.e. (M+1) times, where M is the number of mallocs() required
|
sl@0
|
123 |
# to prepare and execute the statement.
|
sl@0
|
124 |
#
|
sl@0
|
125 |
# Each time an SQL statement fails, the driver program (see proc [run_test]
|
sl@0
|
126 |
# below) figures out if a transaction has been automatically rolled back.
|
sl@0
|
127 |
# If not, it executes any TEST block immediately proceeding the SQL
|
sl@0
|
128 |
# statement, then reexecutes the SQL statement with the next value of N.
|
sl@0
|
129 |
#
|
sl@0
|
130 |
# If a transaction has been automatically rolled back, then the driver
|
sl@0
|
131 |
# program executes all the SQL specified as part of SQL or PREP primitives
|
sl@0
|
132 |
# between the current SQL statement and the most recent "BEGIN". Any
|
sl@0
|
133 |
# TEST block immediately proceeding the SQL statement is evaluated, and
|
sl@0
|
134 |
# then the SQL statement reexecuted with the incremented N value.
|
sl@0
|
135 |
#
|
sl@0
|
136 |
# That make any sense? If not, read the code in [run_test] and it might.
|
sl@0
|
137 |
#
|
sl@0
|
138 |
# Extra restriction imposed by the implementation:
|
sl@0
|
139 |
#
|
sl@0
|
140 |
# * If a PREP block starts a transaction, it must finish it.
|
sl@0
|
141 |
# * A PREP block may not close a transaction it did not start.
|
sl@0
|
142 |
#
|
sl@0
|
143 |
#--------------------------------------------------------------------------
|
sl@0
|
144 |
|
sl@0
|
145 |
# These procs are used to build up a "program" in global variable
|
sl@0
|
146 |
# ::run_test_script. At the end of this file, the proc [run_test] is used
|
sl@0
|
147 |
# to execute the program (and all test cases contained therein).
|
sl@0
|
148 |
#
|
sl@0
|
149 |
set ::run_test_script [list]
|
sl@0
|
150 |
proc TEST {id t} {lappend ::run_test_script -test [list $id $t]}
|
sl@0
|
151 |
proc PREP {p} {lappend ::run_test_script -prep [string trim $p]}
|
sl@0
|
152 |
proc DEBUG {s} {lappend ::run_test_script -debug $s}
|
sl@0
|
153 |
|
sl@0
|
154 |
# SQL --
|
sl@0
|
155 |
#
|
sl@0
|
156 |
# SQL ?-norollback? <sql-text>
|
sl@0
|
157 |
#
|
sl@0
|
158 |
# Add an 'SQL' primitive to the program (see notes above). If the -norollback
|
sl@0
|
159 |
# switch is present, then the statement is not allowed to automatically roll
|
sl@0
|
160 |
# back any active transaction if malloc() fails. It must rollback the statement
|
sl@0
|
161 |
# transaction only.
|
sl@0
|
162 |
#
|
sl@0
|
163 |
proc SQL {a1 {a2 ""}} {
|
sl@0
|
164 |
# An SQL primitive parameter is a list of two elements, a boolean value
|
sl@0
|
165 |
# indicating if the statement may cause transaction rollback when malloc()
|
sl@0
|
166 |
# fails, and the sql statement itself.
|
sl@0
|
167 |
if {$a2 == ""} {
|
sl@0
|
168 |
lappend ::run_test_script -sql [list true [string trim $a1]]
|
sl@0
|
169 |
} else {
|
sl@0
|
170 |
lappend ::run_test_script -sql [list false [string trim $a2]]
|
sl@0
|
171 |
}
|
sl@0
|
172 |
}
|
sl@0
|
173 |
|
sl@0
|
174 |
# TEST_AUTOCOMMIT --
|
sl@0
|
175 |
#
|
sl@0
|
176 |
# A shorthand test to see if a transaction is active or not. The first
|
sl@0
|
177 |
# argument - $id - is the integer number of the test case. The second
|
sl@0
|
178 |
# argument is either 1 or 0, the expected value of the auto-commit flag.
|
sl@0
|
179 |
#
|
sl@0
|
180 |
proc TEST_AUTOCOMMIT {id a} {
|
sl@0
|
181 |
TEST $id "do_test \$testid { sqlite3_get_autocommit \$::DB } {$a}"
|
sl@0
|
182 |
}
|
sl@0
|
183 |
|
sl@0
|
184 |
#--------------------------------------------------------------------------
|
sl@0
|
185 |
# Start of test program declaration
|
sl@0
|
186 |
#
|
sl@0
|
187 |
|
sl@0
|
188 |
|
sl@0
|
189 |
# Warm body test. A malloc() fails in the middle of a CREATE TABLE statement
|
sl@0
|
190 |
# in a single-statement transaction on an empty database. Not too much can go
|
sl@0
|
191 |
# wrong here.
|
sl@0
|
192 |
#
|
sl@0
|
193 |
TEST 1 {
|
sl@0
|
194 |
do_test $testid {
|
sl@0
|
195 |
execsql {SELECT tbl_name FROM sqlite_master;}
|
sl@0
|
196 |
} {}
|
sl@0
|
197 |
}
|
sl@0
|
198 |
SQL {
|
sl@0
|
199 |
CREATE TABLE abc(a, b, c);
|
sl@0
|
200 |
}
|
sl@0
|
201 |
TEST 2 {
|
sl@0
|
202 |
do_test $testid.1 {
|
sl@0
|
203 |
execsql {SELECT tbl_name FROM sqlite_master;}
|
sl@0
|
204 |
} {abc}
|
sl@0
|
205 |
}
|
sl@0
|
206 |
|
sl@0
|
207 |
# Insert a couple of rows into the table. each insert is in its own
|
sl@0
|
208 |
# transaction. test that the table is unpopulated before running the inserts
|
sl@0
|
209 |
# (and hence after each failure of the first insert), and that it has been
|
sl@0
|
210 |
# populated correctly after the final insert succeeds.
|
sl@0
|
211 |
#
|
sl@0
|
212 |
TEST 3 {
|
sl@0
|
213 |
do_test $testid.2 {
|
sl@0
|
214 |
execsql {SELECT * FROM abc}
|
sl@0
|
215 |
} {}
|
sl@0
|
216 |
}
|
sl@0
|
217 |
SQL {INSERT INTO abc VALUES(1, 2, 3);}
|
sl@0
|
218 |
SQL {INSERT INTO abc VALUES(4, 5, 6);}
|
sl@0
|
219 |
SQL {INSERT INTO abc VALUES(7, 8, 9);}
|
sl@0
|
220 |
TEST 4 {
|
sl@0
|
221 |
do_test $testid {
|
sl@0
|
222 |
execsql {SELECT * FROM abc}
|
sl@0
|
223 |
} {1 2 3 4 5 6 7 8 9}
|
sl@0
|
224 |
}
|
sl@0
|
225 |
|
sl@0
|
226 |
# Test a CREATE INDEX statement. Because the table 'abc' is so small, the index
|
sl@0
|
227 |
# will all fit on a single page, so this doesn't test too much that the CREATE
|
sl@0
|
228 |
# TABLE statement didn't test. A few of the transient malloc()s in btree.c
|
sl@0
|
229 |
# perhaps.
|
sl@0
|
230 |
#
|
sl@0
|
231 |
SQL {CREATE INDEX abc_i ON abc(a, b, c);}
|
sl@0
|
232 |
TEST 4 {
|
sl@0
|
233 |
do_test $testid {
|
sl@0
|
234 |
execsql {
|
sl@0
|
235 |
SELECT * FROM abc ORDER BY a DESC;
|
sl@0
|
236 |
}
|
sl@0
|
237 |
} {7 8 9 4 5 6 1 2 3}
|
sl@0
|
238 |
}
|
sl@0
|
239 |
|
sl@0
|
240 |
# Test a DELETE statement. Also create a trigger and a view, just to make sure
|
sl@0
|
241 |
# these statements don't have any obvious malloc() related bugs in them. Note
|
sl@0
|
242 |
# that the test above will be executed each time the DELETE fails, so we're
|
sl@0
|
243 |
# also testing rollback of a DELETE from a table with an index on it.
|
sl@0
|
244 |
#
|
sl@0
|
245 |
SQL {DELETE FROM abc WHERE a > 2;}
|
sl@0
|
246 |
SQL {CREATE TRIGGER abc_t AFTER INSERT ON abc BEGIN SELECT 'trigger!'; END;}
|
sl@0
|
247 |
SQL {CREATE VIEW abc_v AS SELECT * FROM abc;}
|
sl@0
|
248 |
TEST 5 {
|
sl@0
|
249 |
do_test $testid {
|
sl@0
|
250 |
execsql {
|
sl@0
|
251 |
SELECT name, tbl_name FROM sqlite_master ORDER BY name;
|
sl@0
|
252 |
SELECT * FROM abc;
|
sl@0
|
253 |
}
|
sl@0
|
254 |
} {abc abc abc_i abc abc_t abc abc_v abc_v 1 2 3}
|
sl@0
|
255 |
}
|
sl@0
|
256 |
|
sl@0
|
257 |
set sql {
|
sl@0
|
258 |
BEGIN;DELETE FROM abc;
|
sl@0
|
259 |
}
|
sl@0
|
260 |
for {set i 1} {$i < 15} {incr i} {
|
sl@0
|
261 |
set a $i
|
sl@0
|
262 |
set b "String value $i"
|
sl@0
|
263 |
set c [string repeat X $i]
|
sl@0
|
264 |
append sql "INSERT INTO abc VALUES ($a, '$b', '$c');"
|
sl@0
|
265 |
}
|
sl@0
|
266 |
append sql {COMMIT;}
|
sl@0
|
267 |
PREP $sql
|
sl@0
|
268 |
|
sl@0
|
269 |
SQL {
|
sl@0
|
270 |
DELETE FROM abc WHERE oid IN (SELECT oid FROM abc ORDER BY random() LIMIT 5);
|
sl@0
|
271 |
}
|
sl@0
|
272 |
TEST 6 {
|
sl@0
|
273 |
do_test $testid.1 {
|
sl@0
|
274 |
execsql {SELECT count(*) FROM abc}
|
sl@0
|
275 |
} {94}
|
sl@0
|
276 |
do_test $testid.2 {
|
sl@0
|
277 |
execsql {
|
sl@0
|
278 |
SELECT min(
|
sl@0
|
279 |
(oid == a) AND 'String value ' || a == b AND a == length(c)
|
sl@0
|
280 |
) FROM abc;
|
sl@0
|
281 |
}
|
sl@0
|
282 |
} {1}
|
sl@0
|
283 |
}
|
sl@0
|
284 |
SQL {
|
sl@0
|
285 |
DELETE FROM abc WHERE oid IN (SELECT oid FROM abc ORDER BY random() LIMIT 5);
|
sl@0
|
286 |
}
|
sl@0
|
287 |
TEST 7 {
|
sl@0
|
288 |
do_test $testid {
|
sl@0
|
289 |
execsql {SELECT count(*) FROM abc}
|
sl@0
|
290 |
} {89}
|
sl@0
|
291 |
do_test $testid {
|
sl@0
|
292 |
execsql {
|
sl@0
|
293 |
SELECT min(
|
sl@0
|
294 |
(oid == a) AND 'String value ' || a == b AND a == length(c)
|
sl@0
|
295 |
) FROM abc;
|
sl@0
|
296 |
}
|
sl@0
|
297 |
} {1}
|
sl@0
|
298 |
}
|
sl@0
|
299 |
SQL {
|
sl@0
|
300 |
DELETE FROM abc WHERE oid IN (SELECT oid FROM abc ORDER BY random() LIMIT 5);
|
sl@0
|
301 |
}
|
sl@0
|
302 |
TEST 9 {
|
sl@0
|
303 |
do_test $testid {
|
sl@0
|
304 |
execsql {SELECT count(*) FROM abc}
|
sl@0
|
305 |
} {84}
|
sl@0
|
306 |
do_test $testid {
|
sl@0
|
307 |
execsql {
|
sl@0
|
308 |
SELECT min(
|
sl@0
|
309 |
(oid == a) AND 'String value ' || a == b AND a == length(c)
|
sl@0
|
310 |
) FROM abc;
|
sl@0
|
311 |
}
|
sl@0
|
312 |
} {1}
|
sl@0
|
313 |
}
|
sl@0
|
314 |
|
sl@0
|
315 |
set padding [string repeat X 500]
|
sl@0
|
316 |
PREP [subst {
|
sl@0
|
317 |
DROP TABLE abc;
|
sl@0
|
318 |
CREATE TABLE abc(a PRIMARY KEY, padding, b, c);
|
sl@0
|
319 |
INSERT INTO abc VALUES(0, '$padding', 2, 2);
|
sl@0
|
320 |
INSERT INTO abc VALUES(3, '$padding', 5, 5);
|
sl@0
|
321 |
INSERT INTO abc VALUES(6, '$padding', 8, 8);
|
sl@0
|
322 |
}]
|
sl@0
|
323 |
|
sl@0
|
324 |
TEST 10 {
|
sl@0
|
325 |
do_test $testid {
|
sl@0
|
326 |
execsql {SELECT a, b, c FROM abc}
|
sl@0
|
327 |
} {0 2 2 3 5 5 6 8 8}
|
sl@0
|
328 |
}
|
sl@0
|
329 |
|
sl@0
|
330 |
SQL {BEGIN;}
|
sl@0
|
331 |
SQL {INSERT INTO abc VALUES(9, 'XXXXX', 11, 12);}
|
sl@0
|
332 |
TEST_AUTOCOMMIT 11 0
|
sl@0
|
333 |
SQL -norollback {UPDATE abc SET a = a + 1, c = c + 1;}
|
sl@0
|
334 |
TEST_AUTOCOMMIT 12 0
|
sl@0
|
335 |
SQL {DELETE FROM abc WHERE a = 10;}
|
sl@0
|
336 |
TEST_AUTOCOMMIT 13 0
|
sl@0
|
337 |
SQL {COMMIT;}
|
sl@0
|
338 |
|
sl@0
|
339 |
TEST 14 {
|
sl@0
|
340 |
do_test $testid.1 {
|
sl@0
|
341 |
sqlite3_get_autocommit $::DB
|
sl@0
|
342 |
} {1}
|
sl@0
|
343 |
do_test $testid.2 {
|
sl@0
|
344 |
execsql {SELECT a, b, c FROM abc}
|
sl@0
|
345 |
} {1 2 3 4 5 6 7 8 9}
|
sl@0
|
346 |
}
|
sl@0
|
347 |
|
sl@0
|
348 |
PREP [subst {
|
sl@0
|
349 |
DROP TABLE abc;
|
sl@0
|
350 |
CREATE TABLE abc(a, padding, b, c);
|
sl@0
|
351 |
INSERT INTO abc VALUES(1, '$padding', 2, 3);
|
sl@0
|
352 |
INSERT INTO abc VALUES(4, '$padding', 5, 6);
|
sl@0
|
353 |
INSERT INTO abc VALUES(7, '$padding', 8, 9);
|
sl@0
|
354 |
CREATE INDEX abc_i ON abc(a, padding, b, c);
|
sl@0
|
355 |
}]
|
sl@0
|
356 |
|
sl@0
|
357 |
TEST 15 {
|
sl@0
|
358 |
db eval {PRAGMA cache_size = 10}
|
sl@0
|
359 |
}
|
sl@0
|
360 |
|
sl@0
|
361 |
SQL {BEGIN;}
|
sl@0
|
362 |
SQL -norllbck {INSERT INTO abc (oid, a, padding, b, c) SELECT NULL, * FROM abc}
|
sl@0
|
363 |
TEST 16 {
|
sl@0
|
364 |
do_test $testid {
|
sl@0
|
365 |
execsql {SELECT a, count(*) FROM abc GROUP BY a;}
|
sl@0
|
366 |
} {1 2 4 2 7 2}
|
sl@0
|
367 |
}
|
sl@0
|
368 |
SQL -norllbck {INSERT INTO abc (oid, a, padding, b, c) SELECT NULL, * FROM abc}
|
sl@0
|
369 |
TEST 17 {
|
sl@0
|
370 |
do_test $testid {
|
sl@0
|
371 |
execsql {SELECT a, count(*) FROM abc GROUP BY a;}
|
sl@0
|
372 |
} {1 4 4 4 7 4}
|
sl@0
|
373 |
}
|
sl@0
|
374 |
SQL -norllbck {INSERT INTO abc (oid, a, padding, b, c) SELECT NULL, * FROM abc}
|
sl@0
|
375 |
TEST 18 {
|
sl@0
|
376 |
do_test $testid {
|
sl@0
|
377 |
execsql {SELECT a, count(*) FROM abc GROUP BY a;}
|
sl@0
|
378 |
} {1 8 4 8 7 8}
|
sl@0
|
379 |
}
|
sl@0
|
380 |
SQL -norllbck {INSERT INTO abc (oid, a, padding, b, c) SELECT NULL, * FROM abc}
|
sl@0
|
381 |
TEST 19 {
|
sl@0
|
382 |
do_test $testid {
|
sl@0
|
383 |
execsql {SELECT a, count(*) FROM abc GROUP BY a;}
|
sl@0
|
384 |
} {1 16 4 16 7 16}
|
sl@0
|
385 |
}
|
sl@0
|
386 |
SQL {COMMIT;}
|
sl@0
|
387 |
TEST 21 {
|
sl@0
|
388 |
do_test $testid {
|
sl@0
|
389 |
execsql {SELECT a, count(*) FROM abc GROUP BY a;}
|
sl@0
|
390 |
} {1 16 4 16 7 16}
|
sl@0
|
391 |
}
|
sl@0
|
392 |
|
sl@0
|
393 |
SQL {BEGIN;}
|
sl@0
|
394 |
SQL {DELETE FROM abc WHERE oid %2}
|
sl@0
|
395 |
TEST 22 {
|
sl@0
|
396 |
do_test $testid {
|
sl@0
|
397 |
execsql {SELECT a, count(*) FROM abc GROUP BY a;}
|
sl@0
|
398 |
} {1 8 4 8 7 8}
|
sl@0
|
399 |
}
|
sl@0
|
400 |
SQL {DELETE FROM abc}
|
sl@0
|
401 |
TEST 23 {
|
sl@0
|
402 |
do_test $testid {
|
sl@0
|
403 |
execsql {SELECT * FROM abc}
|
sl@0
|
404 |
} {}
|
sl@0
|
405 |
}
|
sl@0
|
406 |
SQL {ROLLBACK;}
|
sl@0
|
407 |
TEST 24 {
|
sl@0
|
408 |
do_test $testid {
|
sl@0
|
409 |
execsql {SELECT a, count(*) FROM abc GROUP BY a;}
|
sl@0
|
410 |
} {1 16 4 16 7 16}
|
sl@0
|
411 |
}
|
sl@0
|
412 |
|
sl@0
|
413 |
# Test some schema modifications inside of a transaction. These should all
|
sl@0
|
414 |
# cause transaction rollback if they fail. Also query a view, to cover a bit
|
sl@0
|
415 |
# more code.
|
sl@0
|
416 |
#
|
sl@0
|
417 |
PREP {DROP VIEW abc_v;}
|
sl@0
|
418 |
TEST 25 {
|
sl@0
|
419 |
do_test $testid {
|
sl@0
|
420 |
execsql {
|
sl@0
|
421 |
SELECT name, tbl_name FROM sqlite_master;
|
sl@0
|
422 |
}
|
sl@0
|
423 |
} {abc abc abc_i abc}
|
sl@0
|
424 |
}
|
sl@0
|
425 |
SQL {BEGIN;}
|
sl@0
|
426 |
SQL {CREATE TABLE def(d, e, f);}
|
sl@0
|
427 |
SQL {CREATE TABLE ghi(g, h, i);}
|
sl@0
|
428 |
TEST 26 {
|
sl@0
|
429 |
do_test $testid {
|
sl@0
|
430 |
execsql {
|
sl@0
|
431 |
SELECT name, tbl_name FROM sqlite_master;
|
sl@0
|
432 |
}
|
sl@0
|
433 |
} {abc abc abc_i abc def def ghi ghi}
|
sl@0
|
434 |
}
|
sl@0
|
435 |
SQL {CREATE VIEW v1 AS SELECT * FROM def, ghi}
|
sl@0
|
436 |
SQL {CREATE UNIQUE INDEX ghi_i1 ON ghi(g);}
|
sl@0
|
437 |
TEST 27 {
|
sl@0
|
438 |
do_test $testid {
|
sl@0
|
439 |
execsql {
|
sl@0
|
440 |
SELECT name, tbl_name FROM sqlite_master;
|
sl@0
|
441 |
}
|
sl@0
|
442 |
} {abc abc abc_i abc def def ghi ghi v1 v1 ghi_i1 ghi}
|
sl@0
|
443 |
}
|
sl@0
|
444 |
SQL {INSERT INTO def VALUES('a', 'b', 'c')}
|
sl@0
|
445 |
SQL {INSERT INTO def VALUES(1, 2, 3)}
|
sl@0
|
446 |
SQL -norollback {INSERT INTO ghi SELECT * FROM def}
|
sl@0
|
447 |
TEST 28 {
|
sl@0
|
448 |
do_test $testid {
|
sl@0
|
449 |
execsql {
|
sl@0
|
450 |
SELECT * FROM def, ghi WHERE d = g;
|
sl@0
|
451 |
}
|
sl@0
|
452 |
} {a b c a b c 1 2 3 1 2 3}
|
sl@0
|
453 |
}
|
sl@0
|
454 |
SQL {COMMIT}
|
sl@0
|
455 |
TEST 29 {
|
sl@0
|
456 |
do_test $testid {
|
sl@0
|
457 |
execsql {
|
sl@0
|
458 |
SELECT * FROM v1 WHERE d = g;
|
sl@0
|
459 |
}
|
sl@0
|
460 |
} {a b c a b c 1 2 3 1 2 3}
|
sl@0
|
461 |
}
|
sl@0
|
462 |
|
sl@0
|
463 |
# Test a simple multi-file transaction
|
sl@0
|
464 |
#
|
sl@0
|
465 |
file delete -force test2.db
|
sl@0
|
466 |
ifcapable attach {
|
sl@0
|
467 |
SQL {ATTACH 'test2.db' AS aux;}
|
sl@0
|
468 |
SQL {BEGIN}
|
sl@0
|
469 |
SQL {CREATE TABLE aux.tbl2(x, y, z)}
|
sl@0
|
470 |
SQL {INSERT INTO tbl2 VALUES(1, 2, 3)}
|
sl@0
|
471 |
SQL {INSERT INTO def VALUES(4, 5, 6)}
|
sl@0
|
472 |
TEST 30 {
|
sl@0
|
473 |
do_test $testid {
|
sl@0
|
474 |
execsql {
|
sl@0
|
475 |
SELECT * FROM tbl2, def WHERE d = x;
|
sl@0
|
476 |
}
|
sl@0
|
477 |
} {1 2 3 1 2 3}
|
sl@0
|
478 |
}
|
sl@0
|
479 |
SQL {COMMIT}
|
sl@0
|
480 |
TEST 31 {
|
sl@0
|
481 |
do_test $testid {
|
sl@0
|
482 |
execsql {
|
sl@0
|
483 |
SELECT * FROM tbl2, def WHERE d = x;
|
sl@0
|
484 |
}
|
sl@0
|
485 |
} {1 2 3 1 2 3}
|
sl@0
|
486 |
}
|
sl@0
|
487 |
}
|
sl@0
|
488 |
|
sl@0
|
489 |
# Test what happens when a malloc() fails while there are other active
|
sl@0
|
490 |
# statements. This changes the way sqlite3VdbeHalt() works.
|
sl@0
|
491 |
TEST 32 {
|
sl@0
|
492 |
if {![info exists ::STMT32]} {
|
sl@0
|
493 |
set sql "SELECT name FROM sqlite_master"
|
sl@0
|
494 |
set ::STMT32 [sqlite3_prepare $::DB $sql -1 DUMMY]
|
sl@0
|
495 |
do_test $testid {
|
sl@0
|
496 |
sqlite3_step $::STMT32
|
sl@0
|
497 |
} {SQLITE_ROW}
|
sl@0
|
498 |
}
|
sl@0
|
499 |
}
|
sl@0
|
500 |
SQL BEGIN
|
sl@0
|
501 |
TEST 33 {
|
sl@0
|
502 |
do_test $testid {
|
sl@0
|
503 |
execsql {SELECT * FROM ghi}
|
sl@0
|
504 |
} {a b c 1 2 3}
|
sl@0
|
505 |
}
|
sl@0
|
506 |
SQL -norollback {
|
sl@0
|
507 |
-- There is a unique index on ghi(g), so this statement may not cause
|
sl@0
|
508 |
-- an automatic ROLLBACK. Hence the "-norollback" switch.
|
sl@0
|
509 |
INSERT INTO ghi SELECT '2'||g, h, i FROM ghi;
|
sl@0
|
510 |
}
|
sl@0
|
511 |
TEST 34 {
|
sl@0
|
512 |
if {[info exists ::STMT32]} {
|
sl@0
|
513 |
do_test $testid {
|
sl@0
|
514 |
sqlite3_finalize $::STMT32
|
sl@0
|
515 |
} {SQLITE_OK}
|
sl@0
|
516 |
unset ::STMT32
|
sl@0
|
517 |
}
|
sl@0
|
518 |
}
|
sl@0
|
519 |
SQL COMMIT
|
sl@0
|
520 |
|
sl@0
|
521 |
#
|
sl@0
|
522 |
# End of test program declaration
|
sl@0
|
523 |
#--------------------------------------------------------------------------
|
sl@0
|
524 |
|
sl@0
|
525 |
proc run_test {arglist iRepeat {pcstart 0} {iFailStart 1}} {
|
sl@0
|
526 |
if {[llength $arglist] %2} {
|
sl@0
|
527 |
error "Uneven number of arguments to TEST"
|
sl@0
|
528 |
}
|
sl@0
|
529 |
|
sl@0
|
530 |
for {set i 0} {$i < $pcstart} {incr i} {
|
sl@0
|
531 |
set k2 [lindex $arglist [expr 2 * $i]]
|
sl@0
|
532 |
set v2 [lindex $arglist [expr 2 * $i + 1]]
|
sl@0
|
533 |
set ac [sqlite3_get_autocommit $::DB] ;# Auto-Commit
|
sl@0
|
534 |
switch -- $k2 {
|
sl@0
|
535 |
-sql {db eval [lindex $v2 1]}
|
sl@0
|
536 |
-prep {db eval $v2}
|
sl@0
|
537 |
}
|
sl@0
|
538 |
set nac [sqlite3_get_autocommit $::DB] ;# New Auto-Commit
|
sl@0
|
539 |
if {$ac && !$nac} {set begin_pc $i}
|
sl@0
|
540 |
}
|
sl@0
|
541 |
|
sl@0
|
542 |
db rollback_hook [list incr ::rollback_hook_count]
|
sl@0
|
543 |
|
sl@0
|
544 |
set iFail $iFailStart
|
sl@0
|
545 |
set pc $pcstart
|
sl@0
|
546 |
while {$pc*2 < [llength $arglist]} {
|
sl@0
|
547 |
|
sl@0
|
548 |
# Id of this iteration:
|
sl@0
|
549 |
set k [lindex $arglist [expr 2 * $pc]]
|
sl@0
|
550 |
set iterid "pc=$pc.iFail=$iFail$k"
|
sl@0
|
551 |
set v [lindex $arglist [expr 2 * $pc + 1]]
|
sl@0
|
552 |
|
sl@0
|
553 |
puts $iterid
|
sl@0
|
554 |
flush stdout
|
sl@0
|
555 |
|
sl@0
|
556 |
switch -- $k {
|
sl@0
|
557 |
|
sl@0
|
558 |
-test {
|
sl@0
|
559 |
foreach {id script} $v {}
|
sl@0
|
560 |
incr pc
|
sl@0
|
561 |
}
|
sl@0
|
562 |
|
sl@0
|
563 |
-sql {
|
sl@0
|
564 |
set ::rollback_hook_count 0
|
sl@0
|
565 |
|
sl@0
|
566 |
set ac [sqlite3_get_autocommit $::DB] ;# Auto-Commit
|
sl@0
|
567 |
sqlite3_memdebug_fail $iFail -repeat 0
|
sl@0
|
568 |
set rc [catch {db eval [lindex $v 1]} msg] ;# True error occurs
|
sl@0
|
569 |
set nac [sqlite3_get_autocommit $::DB] ;# New Auto-Commit
|
sl@0
|
570 |
|
sl@0
|
571 |
if {$rc != 0 && $nac && !$ac} {
|
sl@0
|
572 |
# Before [db eval] the auto-commit flag was clear. Now it
|
sl@0
|
573 |
# is set. Since an error occured we assume this was not a
|
sl@0
|
574 |
# commit - therefore a rollback occured. Check that the
|
sl@0
|
575 |
# rollback-hook was invoked.
|
sl@0
|
576 |
do_test malloc3-rollback_hook.$iterid {
|
sl@0
|
577 |
set ::rollback_hook_count
|
sl@0
|
578 |
} {1}
|
sl@0
|
579 |
}
|
sl@0
|
580 |
|
sl@0
|
581 |
set nFail [sqlite3_memdebug_fail -1 -benigncnt nBenign]
|
sl@0
|
582 |
if {$rc == 0} {
|
sl@0
|
583 |
# Successful execution of sql. The number of failed malloc()
|
sl@0
|
584 |
# calls should be equal to the number of benign failures.
|
sl@0
|
585 |
# Otherwise a malloc() failed and the error was not reported.
|
sl@0
|
586 |
#
|
sl@0
|
587 |
if {$nFail!=$nBenign} {
|
sl@0
|
588 |
error "Unreported malloc() failure"
|
sl@0
|
589 |
}
|
sl@0
|
590 |
|
sl@0
|
591 |
if {$ac && !$nac} {
|
sl@0
|
592 |
# Before the [db eval] the auto-commit flag was set, now it
|
sl@0
|
593 |
# is clear. We can deduce that a "BEGIN" statement has just
|
sl@0
|
594 |
# been successfully executed.
|
sl@0
|
595 |
set begin_pc $pc
|
sl@0
|
596 |
}
|
sl@0
|
597 |
|
sl@0
|
598 |
incr pc
|
sl@0
|
599 |
set iFail 1
|
sl@0
|
600 |
integrity_check "malloc3-(integrity).$iterid"
|
sl@0
|
601 |
} elseif {[regexp {.*out of memory} $msg] || [db errorcode] == 3082} {
|
sl@0
|
602 |
# Out of memory error, as expected.
|
sl@0
|
603 |
#
|
sl@0
|
604 |
integrity_check "malloc3-(integrity).$iterid"
|
sl@0
|
605 |
incr iFail
|
sl@0
|
606 |
if {$nac && !$ac} {
|
sl@0
|
607 |
|
sl@0
|
608 |
if {![lindex $v 0] && [db errorcode] != 3082} {
|
sl@0
|
609 |
# error "Statement \"[lindex $v 1]\" caused a rollback"
|
sl@0
|
610 |
}
|
sl@0
|
611 |
|
sl@0
|
612 |
for {set i $begin_pc} {$i < $pc} {incr i} {
|
sl@0
|
613 |
set k2 [lindex $arglist [expr 2 * $i]]
|
sl@0
|
614 |
set v2 [lindex $arglist [expr 2 * $i + 1]]
|
sl@0
|
615 |
set catchupsql ""
|
sl@0
|
616 |
switch -- $k2 {
|
sl@0
|
617 |
-sql {set catchupsql [lindex $v2 1]}
|
sl@0
|
618 |
-prep {set catchupsql $v2}
|
sl@0
|
619 |
}
|
sl@0
|
620 |
db eval $catchupsql
|
sl@0
|
621 |
}
|
sl@0
|
622 |
}
|
sl@0
|
623 |
} else {
|
sl@0
|
624 |
error $msg
|
sl@0
|
625 |
}
|
sl@0
|
626 |
|
sl@0
|
627 |
while {[lindex $arglist [expr 2 * ($pc -1)]] == "-test"} {
|
sl@0
|
628 |
incr pc -1
|
sl@0
|
629 |
}
|
sl@0
|
630 |
}
|
sl@0
|
631 |
|
sl@0
|
632 |
-prep {
|
sl@0
|
633 |
db eval $v
|
sl@0
|
634 |
incr pc
|
sl@0
|
635 |
}
|
sl@0
|
636 |
|
sl@0
|
637 |
-debug {
|
sl@0
|
638 |
eval $v
|
sl@0
|
639 |
incr pc
|
sl@0
|
640 |
}
|
sl@0
|
641 |
|
sl@0
|
642 |
default { error "Unknown switch: $k" }
|
sl@0
|
643 |
}
|
sl@0
|
644 |
}
|
sl@0
|
645 |
}
|
sl@0
|
646 |
|
sl@0
|
647 |
# Turn of the Tcl interface's prepared statement caching facility. Then
|
sl@0
|
648 |
# run the tests with "persistent" malloc failures.
|
sl@0
|
649 |
sqlite3_extended_result_codes db 1
|
sl@0
|
650 |
db cache size 0
|
sl@0
|
651 |
run_test $::run_test_script 1
|
sl@0
|
652 |
|
sl@0
|
653 |
# Close and reopen the db.
|
sl@0
|
654 |
db close
|
sl@0
|
655 |
file delete -force test.db test.db-journal test2.db test2.db-journal
|
sl@0
|
656 |
sqlite3 db test.db
|
sl@0
|
657 |
sqlite3_extended_result_codes db 1
|
sl@0
|
658 |
set ::DB [sqlite3_connection_pointer db]
|
sl@0
|
659 |
|
sl@0
|
660 |
# Turn of the Tcl interface's prepared statement caching facility in
|
sl@0
|
661 |
# the new connnection. Then run the tests with "transient" malloc failures.
|
sl@0
|
662 |
db cache size 0
|
sl@0
|
663 |
run_test $::run_test_script 0
|
sl@0
|
664 |
|
sl@0
|
665 |
sqlite3_memdebug_fail -1
|
sl@0
|
666 |
finish_test
|