sl@0
|
1 |
# 2003 September 6
|
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 |
# This file implements regression tests for SQLite library. The
|
sl@0
|
12 |
# focus of this script is a test to replicate the bug reported by
|
sl@0
|
13 |
# ticket #842.
|
sl@0
|
14 |
#
|
sl@0
|
15 |
# Ticket #842 was a database corruption problem caused by a DELETE that
|
sl@0
|
16 |
# removed an index entry by not the main table entry. To recreate the
|
sl@0
|
17 |
# problem do this:
|
sl@0
|
18 |
#
|
sl@0
|
19 |
# (1) Create a table with an index. Insert some data into that table.
|
sl@0
|
20 |
# (2) Start a query on the table but do not complete the query.
|
sl@0
|
21 |
# (3) Try to delete a single entry from the table.
|
sl@0
|
22 |
#
|
sl@0
|
23 |
# Step 3 will fail because there is still a read cursor on the table.
|
sl@0
|
24 |
# But the database is corrupted by the DELETE. It turns out that the
|
sl@0
|
25 |
# index entry was deleted first, before the table entry. And the index
|
sl@0
|
26 |
# delete worked. Thus an entry was deleted from the index but not from
|
sl@0
|
27 |
# the table.
|
sl@0
|
28 |
#
|
sl@0
|
29 |
# The solution to the problem was to detect that the table is locked
|
sl@0
|
30 |
# before the index entry is deleted.
|
sl@0
|
31 |
#
|
sl@0
|
32 |
# $Id: delete2.test,v 1.8 2008/07/08 15:59:52 danielk1977 Exp $
|
sl@0
|
33 |
#
|
sl@0
|
34 |
|
sl@0
|
35 |
set testdir [file dirname $argv0]
|
sl@0
|
36 |
source $testdir/tester.tcl
|
sl@0
|
37 |
|
sl@0
|
38 |
# Create a table that has an index.
|
sl@0
|
39 |
#
|
sl@0
|
40 |
do_test delete2-1.1 {
|
sl@0
|
41 |
set DB [sqlite3_connection_pointer db]
|
sl@0
|
42 |
execsql {
|
sl@0
|
43 |
CREATE TABLE q(s string, id string, constraint pk_q primary key(id));
|
sl@0
|
44 |
BEGIN;
|
sl@0
|
45 |
INSERT INTO q(s,id) VALUES('hello','id.1');
|
sl@0
|
46 |
INSERT INTO q(s,id) VALUES('goodbye','id.2');
|
sl@0
|
47 |
INSERT INTO q(s,id) VALUES('again','id.3');
|
sl@0
|
48 |
END;
|
sl@0
|
49 |
SELECT * FROM q;
|
sl@0
|
50 |
}
|
sl@0
|
51 |
} {hello id.1 goodbye id.2 again id.3}
|
sl@0
|
52 |
do_test delete2-1.2 {
|
sl@0
|
53 |
execsql {
|
sl@0
|
54 |
SELECT * FROM q WHERE id='id.1';
|
sl@0
|
55 |
}
|
sl@0
|
56 |
} {hello id.1}
|
sl@0
|
57 |
integrity_check delete2-1.3
|
sl@0
|
58 |
|
sl@0
|
59 |
# Start a query on the table. The query should not use the index.
|
sl@0
|
60 |
# Do not complete the query, thus leaving the table locked.
|
sl@0
|
61 |
#
|
sl@0
|
62 |
do_test delete2-1.4 {
|
sl@0
|
63 |
set STMT [sqlite3_prepare $DB {SELECT * FROM q} -1 TAIL]
|
sl@0
|
64 |
sqlite3_step $STMT
|
sl@0
|
65 |
} SQLITE_ROW
|
sl@0
|
66 |
integrity_check delete2-1.5
|
sl@0
|
67 |
|
sl@0
|
68 |
# Try to delete a row from the table while a read is in process.
|
sl@0
|
69 |
# As of 2006-08-16, this is allowed. (It used to fail with SQLITE_LOCKED.)
|
sl@0
|
70 |
#
|
sl@0
|
71 |
do_test delete2-1.6 {
|
sl@0
|
72 |
catchsql {
|
sl@0
|
73 |
DELETE FROM q WHERE rowid=1
|
sl@0
|
74 |
}
|
sl@0
|
75 |
} {0 {}}
|
sl@0
|
76 |
integrity_check delete2-1.7
|
sl@0
|
77 |
do_test delete2-1.8 {
|
sl@0
|
78 |
execsql {
|
sl@0
|
79 |
SELECT * FROM q;
|
sl@0
|
80 |
}
|
sl@0
|
81 |
} {goodbye id.2 again id.3}
|
sl@0
|
82 |
|
sl@0
|
83 |
# Finalize the query, thus clearing the lock on the table. Then
|
sl@0
|
84 |
# retry the delete. The delete should work this time.
|
sl@0
|
85 |
#
|
sl@0
|
86 |
do_test delete2-1.9 {
|
sl@0
|
87 |
sqlite3_finalize $STMT
|
sl@0
|
88 |
catchsql {
|
sl@0
|
89 |
DELETE FROM q WHERE rowid=1
|
sl@0
|
90 |
}
|
sl@0
|
91 |
} {0 {}}
|
sl@0
|
92 |
integrity_check delete2-1.10
|
sl@0
|
93 |
do_test delete2-1.11 {
|
sl@0
|
94 |
execsql {
|
sl@0
|
95 |
SELECT * FROM q;
|
sl@0
|
96 |
}
|
sl@0
|
97 |
} {goodbye id.2 again id.3}
|
sl@0
|
98 |
|
sl@0
|
99 |
do_test delete2-2.1 {
|
sl@0
|
100 |
execsql {
|
sl@0
|
101 |
CREATE TABLE t1(a, b);
|
sl@0
|
102 |
CREATE TABLE t2(c, d);
|
sl@0
|
103 |
INSERT INTO t1 VALUES(1, 2);
|
sl@0
|
104 |
INSERT INTO t2 VALUES(3, 4);
|
sl@0
|
105 |
INSERT INTO t2 VALUES(5, 6);
|
sl@0
|
106 |
}
|
sl@0
|
107 |
} {}
|
sl@0
|
108 |
do_test delete2-2.2 {
|
sl@0
|
109 |
set res [list]
|
sl@0
|
110 |
db eval {
|
sl@0
|
111 |
SELECT CASE WHEN c = 5 THEN b ELSE NULL END AS b, c, d FROM t1, t2
|
sl@0
|
112 |
} {
|
sl@0
|
113 |
db eval {DELETE FROM t1}
|
sl@0
|
114 |
lappend res $b $c $d
|
sl@0
|
115 |
}
|
sl@0
|
116 |
set res
|
sl@0
|
117 |
} {{} 3 4 {} 5 6}
|
sl@0
|
118 |
|
sl@0
|
119 |
finish_test
|