sl@0
|
1 |
# 2005 December 30
|
sl@0
|
2 |
#
|
sl@0
|
3 |
# Portions Copyright (c) 2007-2008 Nokia Corporation and/or its subsidiaries. All rights reserved.
|
sl@0
|
4 |
#
|
sl@0
|
5 |
# The author disclaims copyright to this source code. In place of
|
sl@0
|
6 |
# a legal notice, here is a blessing:
|
sl@0
|
7 |
#
|
sl@0
|
8 |
# May you do good and not evil.
|
sl@0
|
9 |
# May you find forgiveness for yourself and forgive others.
|
sl@0
|
10 |
# May you share freely, never taking more than you give.
|
sl@0
|
11 |
#
|
sl@0
|
12 |
#***********************************************************************
|
sl@0
|
13 |
#
|
sl@0
|
14 |
# $Id: shared.test,v 1.34 2008/07/12 14:52:20 drh Exp $
|
sl@0
|
15 |
|
sl@0
|
16 |
set testdir [file dirname $argv0]
|
sl@0
|
17 |
source $testdir/tester.tcl
|
sl@0
|
18 |
db close
|
sl@0
|
19 |
|
sl@0
|
20 |
# These tests cannot be run without the ATTACH command.
|
sl@0
|
21 |
#
|
sl@0
|
22 |
ifcapable !shared_cache||!attach {
|
sl@0
|
23 |
finish_test
|
sl@0
|
24 |
return
|
sl@0
|
25 |
}
|
sl@0
|
26 |
|
sl@0
|
27 |
set ::enable_shared_cache [sqlite3_enable_shared_cache 1]
|
sl@0
|
28 |
|
sl@0
|
29 |
foreach av [list 0 1] {
|
sl@0
|
30 |
|
sl@0
|
31 |
# Open the database connection and execute the auto-vacuum pragma
|
sl@0
|
32 |
file delete -force test.db
|
sl@0
|
33 |
sqlite3 db test.db
|
sl@0
|
34 |
|
sl@0
|
35 |
ifcapable autovacuum {
|
sl@0
|
36 |
do_test shared-[expr $av+1].1.0 {
|
sl@0
|
37 |
execsql "pragma auto_vacuum=$::av"
|
sl@0
|
38 |
execsql {pragma auto_vacuum}
|
sl@0
|
39 |
} "$av"
|
sl@0
|
40 |
} else {
|
sl@0
|
41 |
if {$av} {
|
sl@0
|
42 |
db close
|
sl@0
|
43 |
break
|
sl@0
|
44 |
}
|
sl@0
|
45 |
}
|
sl@0
|
46 |
|
sl@0
|
47 |
# $av is currently 0 if this loop iteration is to test with auto-vacuum turned
|
sl@0
|
48 |
# off, and 1 if it is turned on. Increment it so that (1 -> no auto-vacuum)
|
sl@0
|
49 |
# and (2 -> auto-vacuum). The sole reason for this is so that it looks nicer
|
sl@0
|
50 |
# when we use this variable as part of test-case names.
|
sl@0
|
51 |
#
|
sl@0
|
52 |
incr av
|
sl@0
|
53 |
|
sl@0
|
54 |
# Test organization:
|
sl@0
|
55 |
#
|
sl@0
|
56 |
# shared-1.*: Simple test to verify basic sanity of table level locking when
|
sl@0
|
57 |
# two connections share a pager cache.
|
sl@0
|
58 |
# shared-2.*: Test that a read transaction can co-exist with a
|
sl@0
|
59 |
# write-transaction, including a simple test to ensure the
|
sl@0
|
60 |
# external locking protocol is still working.
|
sl@0
|
61 |
# shared-3.*: Simple test of read-uncommitted mode.
|
sl@0
|
62 |
# shared-4.*: Check that the schema is locked and unlocked correctly.
|
sl@0
|
63 |
# shared-5.*: Test that creating/dropping schema items works when databases
|
sl@0
|
64 |
# are attached in different orders to different handles.
|
sl@0
|
65 |
# shared-6.*: Locking, UNION ALL queries and sub-queries.
|
sl@0
|
66 |
# shared-7.*: Autovacuum and shared-cache.
|
sl@0
|
67 |
# shared-8.*: Tests related to the text encoding of shared-cache databases.
|
sl@0
|
68 |
# shared-9.*: TEMP triggers and shared-cache databases.
|
sl@0
|
69 |
# shared-10.*: Tests of sqlite3_close().
|
sl@0
|
70 |
# shared-11.*: Test transaction locking.
|
sl@0
|
71 |
#
|
sl@0
|
72 |
|
sl@0
|
73 |
do_test shared-$av.1.1 {
|
sl@0
|
74 |
# Open a second database on the file test.db. It should use the same pager
|
sl@0
|
75 |
# cache and schema as the original connection. Verify that only 1 file is
|
sl@0
|
76 |
# opened.
|
sl@0
|
77 |
sqlite3 db2 test.db
|
sl@0
|
78 |
set ::sqlite_open_file_count
|
sl@0
|
79 |
} {1}
|
sl@0
|
80 |
do_test shared-$av.1.2 {
|
sl@0
|
81 |
# Add a table and a single row of data via the first connection.
|
sl@0
|
82 |
# Ensure that the second connection can see them.
|
sl@0
|
83 |
execsql {
|
sl@0
|
84 |
CREATE TABLE abc(a, b, c);
|
sl@0
|
85 |
INSERT INTO abc VALUES(1, 2, 3);
|
sl@0
|
86 |
} db
|
sl@0
|
87 |
execsql {
|
sl@0
|
88 |
SELECT * FROM abc;
|
sl@0
|
89 |
} db2
|
sl@0
|
90 |
} {1 2 3}
|
sl@0
|
91 |
do_test shared-$av.1.3 {
|
sl@0
|
92 |
# Have the first connection begin a transaction and obtain a read-lock
|
sl@0
|
93 |
# on table abc. This should not prevent the second connection from
|
sl@0
|
94 |
# querying abc.
|
sl@0
|
95 |
execsql {
|
sl@0
|
96 |
BEGIN;
|
sl@0
|
97 |
SELECT * FROM abc;
|
sl@0
|
98 |
}
|
sl@0
|
99 |
execsql {
|
sl@0
|
100 |
SELECT * FROM abc;
|
sl@0
|
101 |
} db2
|
sl@0
|
102 |
} {1 2 3}
|
sl@0
|
103 |
do_test shared-$av.1.4 {
|
sl@0
|
104 |
# Try to insert a row into abc via connection 2. This should fail because
|
sl@0
|
105 |
# of the read-lock connection 1 is holding on table abc (obtained in the
|
sl@0
|
106 |
# previous test case).
|
sl@0
|
107 |
catchsql {
|
sl@0
|
108 |
INSERT INTO abc VALUES(4, 5, 6);
|
sl@0
|
109 |
} db2
|
sl@0
|
110 |
} {1 {database table is locked: abc}}
|
sl@0
|
111 |
do_test shared-$av.1.5 {
|
sl@0
|
112 |
# Using connection 2 (the one without the open transaction), try to create
|
sl@0
|
113 |
# a new table. This should fail because of the open read transaction
|
sl@0
|
114 |
# held by connection 1.
|
sl@0
|
115 |
catchsql {
|
sl@0
|
116 |
CREATE TABLE def(d, e, f);
|
sl@0
|
117 |
} db2
|
sl@0
|
118 |
} {1 {database table is locked: sqlite_master}}
|
sl@0
|
119 |
do_test shared-$av.1.6 {
|
sl@0
|
120 |
# Upgrade connection 1's transaction to a write transaction. Create
|
sl@0
|
121 |
# a new table - def - and insert a row into it. Because the connection 1
|
sl@0
|
122 |
# transaction modifies the schema, it should not be possible for
|
sl@0
|
123 |
# connection 2 to access the database at all until the connection 1
|
sl@0
|
124 |
# has finished the transaction.
|
sl@0
|
125 |
execsql {
|
sl@0
|
126 |
CREATE TABLE def(d, e, f);
|
sl@0
|
127 |
INSERT INTO def VALUES('IV', 'V', 'VI');
|
sl@0
|
128 |
}
|
sl@0
|
129 |
} {}
|
sl@0
|
130 |
do_test shared-$av.1.7 {
|
sl@0
|
131 |
# Read from the sqlite_master table with connection 1 (inside the
|
sl@0
|
132 |
# transaction). Then test that we can not do this with connection 2. This
|
sl@0
|
133 |
# is because of the schema-modified lock established by connection 1
|
sl@0
|
134 |
# in the previous test case.
|
sl@0
|
135 |
execsql {
|
sl@0
|
136 |
SELECT * FROM sqlite_master;
|
sl@0
|
137 |
}
|
sl@0
|
138 |
catchsql {
|
sl@0
|
139 |
SELECT * FROM sqlite_master;
|
sl@0
|
140 |
} db2
|
sl@0
|
141 |
} {1 {database schema is locked: main}}
|
sl@0
|
142 |
do_test shared-$av.1.8 {
|
sl@0
|
143 |
# Commit the connection 1 transaction.
|
sl@0
|
144 |
execsql {
|
sl@0
|
145 |
COMMIT;
|
sl@0
|
146 |
}
|
sl@0
|
147 |
} {}
|
sl@0
|
148 |
|
sl@0
|
149 |
do_test shared-$av.2.1 {
|
sl@0
|
150 |
# Open connection db3 to the database. Use a different path to the same
|
sl@0
|
151 |
# file so that db3 does *not* share the same pager cache as db and db2
|
sl@0
|
152 |
# (there should be two open file handles).
|
sl@0
|
153 |
if {$::tcl_platform(platform)=="unix"} {
|
sl@0
|
154 |
sqlite3 db3 ./test.db
|
sl@0
|
155 |
} else {
|
sl@0
|
156 |
sqlite3 db3 TEST.DB
|
sl@0
|
157 |
}
|
sl@0
|
158 |
set ::sqlite_open_file_count
|
sl@0
|
159 |
} {2}
|
sl@0
|
160 |
do_test shared-$av.2.2 {
|
sl@0
|
161 |
# Start read transactions on db and db2 (the shared pager cache). Ensure
|
sl@0
|
162 |
# db3 cannot write to the database.
|
sl@0
|
163 |
execsql {
|
sl@0
|
164 |
BEGIN;
|
sl@0
|
165 |
SELECT * FROM abc;
|
sl@0
|
166 |
}
|
sl@0
|
167 |
execsql {
|
sl@0
|
168 |
BEGIN;
|
sl@0
|
169 |
SELECT * FROM abc;
|
sl@0
|
170 |
} db2
|
sl@0
|
171 |
catchsql {
|
sl@0
|
172 |
INSERT INTO abc VALUES(1, 2, 3);
|
sl@0
|
173 |
} db2
|
sl@0
|
174 |
} {1 {database table is locked: abc}}
|
sl@0
|
175 |
do_test shared-$av.2.3 {
|
sl@0
|
176 |
# Turn db's transaction into a write-transaction. db3 should still be
|
sl@0
|
177 |
# able to read from table def (but will not see the new row). Connection
|
sl@0
|
178 |
# db2 should not be able to read def (because of the write-lock).
|
sl@0
|
179 |
|
sl@0
|
180 |
# Todo: The failed "INSERT INTO abc ..." statement in the above test
|
sl@0
|
181 |
# has started a write-transaction on db2 (should this be so?). This
|
sl@0
|
182 |
# would prevent connection db from starting a write-transaction. So roll the
|
sl@0
|
183 |
# db2 transaction back and replace it with a new read transaction.
|
sl@0
|
184 |
execsql {
|
sl@0
|
185 |
ROLLBACK;
|
sl@0
|
186 |
BEGIN;
|
sl@0
|
187 |
SELECT * FROM abc;
|
sl@0
|
188 |
} db2
|
sl@0
|
189 |
|
sl@0
|
190 |
execsql {
|
sl@0
|
191 |
INSERT INTO def VALUES('VII', 'VIII', 'IX');
|
sl@0
|
192 |
}
|
sl@0
|
193 |
concat [
|
sl@0
|
194 |
catchsql { SELECT * FROM def; } db3
|
sl@0
|
195 |
] [
|
sl@0
|
196 |
catchsql { SELECT * FROM def; } db2
|
sl@0
|
197 |
]
|
sl@0
|
198 |
} {0 {IV V VI} 1 {database table is locked: def}}
|
sl@0
|
199 |
do_test shared-$av.2.4 {
|
sl@0
|
200 |
# Commit the open transaction on db. db2 still holds a read-transaction.
|
sl@0
|
201 |
# This should prevent db3 from writing to the database, but not from
|
sl@0
|
202 |
# reading.
|
sl@0
|
203 |
execsql {
|
sl@0
|
204 |
COMMIT;
|
sl@0
|
205 |
}
|
sl@0
|
206 |
concat [
|
sl@0
|
207 |
catchsql { SELECT * FROM def; } db3
|
sl@0
|
208 |
] [
|
sl@0
|
209 |
catchsql { INSERT INTO def VALUES('X', 'XI', 'XII'); } db3
|
sl@0
|
210 |
]
|
sl@0
|
211 |
} {0 {IV V VI VII VIII IX} 1 {database is locked}}
|
sl@0
|
212 |
|
sl@0
|
213 |
catchsql COMMIT db2
|
sl@0
|
214 |
|
sl@0
|
215 |
do_test shared-$av.3.1.1 {
|
sl@0
|
216 |
# This test case starts a linear scan of table 'seq' using a
|
sl@0
|
217 |
# read-uncommitted connection. In the middle of the scan, rows are added
|
sl@0
|
218 |
# to the end of the seq table (ahead of the current cursor position).
|
sl@0
|
219 |
# The uncommitted rows should be included in the results of the scan.
|
sl@0
|
220 |
execsql "
|
sl@0
|
221 |
CREATE TABLE seq(i PRIMARY KEY, x);
|
sl@0
|
222 |
INSERT INTO seq VALUES(1, '[string repeat X 500]');
|
sl@0
|
223 |
INSERT INTO seq VALUES(2, '[string repeat X 500]');
|
sl@0
|
224 |
"
|
sl@0
|
225 |
execsql {SELECT * FROM sqlite_master} db2
|
sl@0
|
226 |
execsql {PRAGMA read_uncommitted = 1} db2
|
sl@0
|
227 |
|
sl@0
|
228 |
set ret [list]
|
sl@0
|
229 |
db2 eval {SELECT i FROM seq ORDER BY i} {
|
sl@0
|
230 |
if {$i < 4} {
|
sl@0
|
231 |
set max [execsql {SELECT max(i) FROM seq}]
|
sl@0
|
232 |
db eval {
|
sl@0
|
233 |
INSERT INTO seq SELECT i + :max, x FROM seq;
|
sl@0
|
234 |
}
|
sl@0
|
235 |
}
|
sl@0
|
236 |
lappend ret $i
|
sl@0
|
237 |
}
|
sl@0
|
238 |
set ret
|
sl@0
|
239 |
} {1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16}
|
sl@0
|
240 |
do_test shared-$av.3.1.2 {
|
sl@0
|
241 |
# Another linear scan through table seq using a read-uncommitted connection.
|
sl@0
|
242 |
# This time, delete each row as it is read. Should not affect the results of
|
sl@0
|
243 |
# the scan, but the table should be empty after the scan is concluded
|
sl@0
|
244 |
# (test 3.1.3 verifies this).
|
sl@0
|
245 |
set ret [list]
|
sl@0
|
246 |
db2 eval {SELECT i FROM seq} {
|
sl@0
|
247 |
db eval {DELETE FROM seq WHERE i = :i}
|
sl@0
|
248 |
lappend ret $i
|
sl@0
|
249 |
}
|
sl@0
|
250 |
set ret
|
sl@0
|
251 |
} {1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16}
|
sl@0
|
252 |
do_test shared-$av.3.1.3 {
|
sl@0
|
253 |
execsql {
|
sl@0
|
254 |
SELECT * FROM seq;
|
sl@0
|
255 |
}
|
sl@0
|
256 |
} {}
|
sl@0
|
257 |
|
sl@0
|
258 |
catch {db close}
|
sl@0
|
259 |
catch {db2 close}
|
sl@0
|
260 |
catch {db3 close}
|
sl@0
|
261 |
|
sl@0
|
262 |
#--------------------------------------------------------------------------
|
sl@0
|
263 |
# Tests shared-4.* test that the schema locking rules are applied
|
sl@0
|
264 |
# correctly. i.e.:
|
sl@0
|
265 |
#
|
sl@0
|
266 |
# 1. All transactions require a read-lock on the schemas of databases they
|
sl@0
|
267 |
# access.
|
sl@0
|
268 |
# 2. Transactions that modify a database schema require a write-lock on that
|
sl@0
|
269 |
# schema.
|
sl@0
|
270 |
# 3. It is not possible to compile a statement while another handle has a
|
sl@0
|
271 |
# write-lock on the schema.
|
sl@0
|
272 |
#
|
sl@0
|
273 |
|
sl@0
|
274 |
# Open two database handles db and db2. Each has a single attach database
|
sl@0
|
275 |
# (as well as main):
|
sl@0
|
276 |
#
|
sl@0
|
277 |
# db.main -> ./test.db
|
sl@0
|
278 |
# db.test2 -> ./test2.db
|
sl@0
|
279 |
# db2.main -> ./test2.db
|
sl@0
|
280 |
# db2.test -> ./test.db
|
sl@0
|
281 |
#
|
sl@0
|
282 |
file delete -force test.db
|
sl@0
|
283 |
file delete -force test2.db
|
sl@0
|
284 |
file delete -force test2.db-journal
|
sl@0
|
285 |
sqlite3 db test.db
|
sl@0
|
286 |
sqlite3 db2 test2.db
|
sl@0
|
287 |
do_test shared-$av.4.1.1 {
|
sl@0
|
288 |
set sqlite_open_file_count
|
sl@0
|
289 |
} {2}
|
sl@0
|
290 |
do_test shared-$av.4.1.2 {
|
sl@0
|
291 |
execsql {ATTACH 'test2.db' AS test2}
|
sl@0
|
292 |
set sqlite_open_file_count
|
sl@0
|
293 |
} {2}
|
sl@0
|
294 |
do_test shared-$av.4.1.3 {
|
sl@0
|
295 |
execsql {ATTACH 'test.db' AS test} db2
|
sl@0
|
296 |
set sqlite_open_file_count
|
sl@0
|
297 |
} {2}
|
sl@0
|
298 |
|
sl@0
|
299 |
# Sanity check: Create a table in ./test.db via handle db, and test that handle
|
sl@0
|
300 |
# db2 can "see" the new table immediately. A handle using a seperate pager
|
sl@0
|
301 |
# cache would have to reload the database schema before this were possible.
|
sl@0
|
302 |
#
|
sl@0
|
303 |
do_test shared-$av.4.2.1 {
|
sl@0
|
304 |
execsql {
|
sl@0
|
305 |
CREATE TABLE abc(a, b, c);
|
sl@0
|
306 |
CREATE TABLE def(d, e, f);
|
sl@0
|
307 |
INSERT INTO abc VALUES('i', 'ii', 'iii');
|
sl@0
|
308 |
INSERT INTO def VALUES('I', 'II', 'III');
|
sl@0
|
309 |
}
|
sl@0
|
310 |
} {}
|
sl@0
|
311 |
do_test shared-$av.4.2.2 {
|
sl@0
|
312 |
execsql {
|
sl@0
|
313 |
SELECT * FROM test.abc;
|
sl@0
|
314 |
} db2
|
sl@0
|
315 |
} {i ii iii}
|
sl@0
|
316 |
|
sl@0
|
317 |
# Open a read-transaction and read from table abc via handle 2. Check that
|
sl@0
|
318 |
# handle 1 can read table abc. Check that handle 1 cannot modify table abc
|
sl@0
|
319 |
# or the database schema. Then check that handle 1 can modify table def.
|
sl@0
|
320 |
#
|
sl@0
|
321 |
do_test shared-$av.4.3.1 {
|
sl@0
|
322 |
execsql {
|
sl@0
|
323 |
BEGIN;
|
sl@0
|
324 |
SELECT * FROM test.abc;
|
sl@0
|
325 |
} db2
|
sl@0
|
326 |
} {i ii iii}
|
sl@0
|
327 |
do_test shared-$av.4.3.2 {
|
sl@0
|
328 |
catchsql {
|
sl@0
|
329 |
INSERT INTO abc VALUES('iv', 'v', 'vi');
|
sl@0
|
330 |
}
|
sl@0
|
331 |
} {1 {database table is locked: abc}}
|
sl@0
|
332 |
do_test shared-$av.4.3.3 {
|
sl@0
|
333 |
catchsql {
|
sl@0
|
334 |
CREATE TABLE ghi(g, h, i);
|
sl@0
|
335 |
}
|
sl@0
|
336 |
} {1 {database table is locked: sqlite_master}}
|
sl@0
|
337 |
do_test shared-$av.4.3.3 {
|
sl@0
|
338 |
catchsql {
|
sl@0
|
339 |
INSERT INTO def VALUES('IV', 'V', 'VI');
|
sl@0
|
340 |
}
|
sl@0
|
341 |
} {0 {}}
|
sl@0
|
342 |
do_test shared-$av.4.3.4 {
|
sl@0
|
343 |
# Cleanup: commit the transaction opened by db2.
|
sl@0
|
344 |
execsql {
|
sl@0
|
345 |
COMMIT
|
sl@0
|
346 |
} db2
|
sl@0
|
347 |
} {}
|
sl@0
|
348 |
|
sl@0
|
349 |
# Open a write-transaction using handle 1 and modify the database schema.
|
sl@0
|
350 |
# Then try to execute a compiled statement to read from the same
|
sl@0
|
351 |
# database via handle 2 (fails to get the lock on sqlite_master). Also
|
sl@0
|
352 |
# try to compile a read of the same database using handle 2 (also fails).
|
sl@0
|
353 |
# Finally, compile a read of the other database using handle 2. This
|
sl@0
|
354 |
# should also fail.
|
sl@0
|
355 |
#
|
sl@0
|
356 |
ifcapable compound {
|
sl@0
|
357 |
do_test shared-$av.4.4.1.2 {
|
sl@0
|
358 |
# Sanity check 1: Check that the schema is what we think it is when viewed
|
sl@0
|
359 |
# via handle 1.
|
sl@0
|
360 |
execsql {
|
sl@0
|
361 |
CREATE TABLE test2.ghi(g, h, i);
|
sl@0
|
362 |
SELECT 'test.db:'||name FROM sqlite_master
|
sl@0
|
363 |
UNION ALL
|
sl@0
|
364 |
SELECT 'test2.db:'||name FROM test2.sqlite_master;
|
sl@0
|
365 |
}
|
sl@0
|
366 |
} {test.db:abc test.db:def test2.db:ghi}
|
sl@0
|
367 |
do_test shared-$av.4.4.1.2 {
|
sl@0
|
368 |
# Sanity check 2: Check that the schema is what we think it is when viewed
|
sl@0
|
369 |
# via handle 2.
|
sl@0
|
370 |
execsql {
|
sl@0
|
371 |
SELECT 'test2.db:'||name FROM sqlite_master
|
sl@0
|
372 |
UNION ALL
|
sl@0
|
373 |
SELECT 'test.db:'||name FROM test.sqlite_master;
|
sl@0
|
374 |
} db2
|
sl@0
|
375 |
} {test2.db:ghi test.db:abc test.db:def}
|
sl@0
|
376 |
}
|
sl@0
|
377 |
|
sl@0
|
378 |
do_test shared-$av.4.4.2 {
|
sl@0
|
379 |
set ::DB2 [sqlite3_connection_pointer db2]
|
sl@0
|
380 |
set sql {SELECT * FROM abc}
|
sl@0
|
381 |
set ::STMT1 [sqlite3_prepare $::DB2 $sql -1 DUMMY]
|
sl@0
|
382 |
execsql {
|
sl@0
|
383 |
BEGIN;
|
sl@0
|
384 |
CREATE TABLE jkl(j, k, l);
|
sl@0
|
385 |
}
|
sl@0
|
386 |
sqlite3_step $::STMT1
|
sl@0
|
387 |
} {SQLITE_ERROR}
|
sl@0
|
388 |
do_test shared-$av.4.4.3 {
|
sl@0
|
389 |
sqlite3_finalize $::STMT1
|
sl@0
|
390 |
} {SQLITE_LOCKED}
|
sl@0
|
391 |
do_test shared-$av.4.4.4 {
|
sl@0
|
392 |
set rc [catch {
|
sl@0
|
393 |
set ::STMT1 [sqlite3_prepare $::DB2 $sql -1 DUMMY]
|
sl@0
|
394 |
} msg]
|
sl@0
|
395 |
list $rc $msg
|
sl@0
|
396 |
} {1 {(6) database schema is locked: test}}
|
sl@0
|
397 |
do_test shared-$av.4.4.5 {
|
sl@0
|
398 |
set rc [catch {
|
sl@0
|
399 |
set ::STMT1 [sqlite3_prepare $::DB2 "SELECT * FROM ghi" -1 DUMMY]
|
sl@0
|
400 |
} msg]
|
sl@0
|
401 |
list $rc $msg
|
sl@0
|
402 |
} {1 {(6) database schema is locked: test}}
|
sl@0
|
403 |
|
sl@0
|
404 |
|
sl@0
|
405 |
catch {db2 close}
|
sl@0
|
406 |
catch {db close}
|
sl@0
|
407 |
|
sl@0
|
408 |
#--------------------------------------------------------------------------
|
sl@0
|
409 |
# Tests shared-5.*
|
sl@0
|
410 |
#
|
sl@0
|
411 |
foreach db [list test.db test1.db test2.db test3.db] {
|
sl@0
|
412 |
file delete -force $db ${db}-journal
|
sl@0
|
413 |
}
|
sl@0
|
414 |
do_test shared-$av.5.1.1 {
|
sl@0
|
415 |
sqlite3 db1 test.db
|
sl@0
|
416 |
sqlite3 db2 test.db
|
sl@0
|
417 |
execsql {
|
sl@0
|
418 |
ATTACH 'test1.db' AS test1;
|
sl@0
|
419 |
ATTACH 'test2.db' AS test2;
|
sl@0
|
420 |
ATTACH 'test3.db' AS test3;
|
sl@0
|
421 |
} db1
|
sl@0
|
422 |
execsql {
|
sl@0
|
423 |
ATTACH 'test3.db' AS test3;
|
sl@0
|
424 |
ATTACH 'test2.db' AS test2;
|
sl@0
|
425 |
ATTACH 'test1.db' AS test1;
|
sl@0
|
426 |
} db2
|
sl@0
|
427 |
} {}
|
sl@0
|
428 |
do_test shared-$av.5.1.2 {
|
sl@0
|
429 |
execsql {
|
sl@0
|
430 |
CREATE TABLE test1.t1(a, b);
|
sl@0
|
431 |
CREATE INDEX test1.i1 ON t1(a, b);
|
sl@0
|
432 |
} db1
|
sl@0
|
433 |
} {}
|
sl@0
|
434 |
ifcapable view {
|
sl@0
|
435 |
do_test shared-$av.5.1.3 {
|
sl@0
|
436 |
execsql {
|
sl@0
|
437 |
CREATE VIEW test1.v1 AS SELECT * FROM t1;
|
sl@0
|
438 |
} db1
|
sl@0
|
439 |
} {}
|
sl@0
|
440 |
}
|
sl@0
|
441 |
ifcapable trigger {
|
sl@0
|
442 |
do_test shared-$av.5.1.4 {
|
sl@0
|
443 |
execsql {
|
sl@0
|
444 |
CREATE TRIGGER test1.trig1 AFTER INSERT ON t1 BEGIN
|
sl@0
|
445 |
INSERT INTO t1 VALUES(new.a, new.b);
|
sl@0
|
446 |
END;
|
sl@0
|
447 |
} db1
|
sl@0
|
448 |
} {}
|
sl@0
|
449 |
}
|
sl@0
|
450 |
do_test shared-$av.5.1.5 {
|
sl@0
|
451 |
execsql {
|
sl@0
|
452 |
DROP INDEX i1;
|
sl@0
|
453 |
} db2
|
sl@0
|
454 |
} {}
|
sl@0
|
455 |
ifcapable view {
|
sl@0
|
456 |
do_test shared-$av.5.1.6 {
|
sl@0
|
457 |
execsql {
|
sl@0
|
458 |
DROP VIEW v1;
|
sl@0
|
459 |
} db2
|
sl@0
|
460 |
} {}
|
sl@0
|
461 |
}
|
sl@0
|
462 |
ifcapable trigger {
|
sl@0
|
463 |
do_test shared-$av.5.1.7 {
|
sl@0
|
464 |
execsql {
|
sl@0
|
465 |
DROP TRIGGER trig1;
|
sl@0
|
466 |
} db2
|
sl@0
|
467 |
} {}
|
sl@0
|
468 |
}
|
sl@0
|
469 |
do_test shared-$av.5.1.8 {
|
sl@0
|
470 |
execsql {
|
sl@0
|
471 |
DROP TABLE t1;
|
sl@0
|
472 |
} db2
|
sl@0
|
473 |
} {}
|
sl@0
|
474 |
ifcapable compound {
|
sl@0
|
475 |
do_test shared-$av.5.1.9 {
|
sl@0
|
476 |
execsql {
|
sl@0
|
477 |
SELECT * FROM sqlite_master UNION ALL SELECT * FROM test1.sqlite_master
|
sl@0
|
478 |
} db1
|
sl@0
|
479 |
} {}
|
sl@0
|
480 |
}
|
sl@0
|
481 |
|
sl@0
|
482 |
#--------------------------------------------------------------------------
|
sl@0
|
483 |
# Tests shared-6.* test that a query obtains all the read-locks it needs
|
sl@0
|
484 |
# before starting execution of the query. This means that there is no chance
|
sl@0
|
485 |
# some rows of data will be returned before a lock fails and SQLITE_LOCK
|
sl@0
|
486 |
# is returned.
|
sl@0
|
487 |
#
|
sl@0
|
488 |
do_test shared-$av.6.1.1 {
|
sl@0
|
489 |
execsql {
|
sl@0
|
490 |
CREATE TABLE t1(a, b);
|
sl@0
|
491 |
CREATE TABLE t2(a, b);
|
sl@0
|
492 |
INSERT INTO t1 VALUES(1, 2);
|
sl@0
|
493 |
INSERT INTO t2 VALUES(3, 4);
|
sl@0
|
494 |
} db1
|
sl@0
|
495 |
} {}
|
sl@0
|
496 |
ifcapable compound {
|
sl@0
|
497 |
do_test shared-$av.6.1.2 {
|
sl@0
|
498 |
execsql {
|
sl@0
|
499 |
SELECT * FROM t1 UNION ALL SELECT * FROM t2;
|
sl@0
|
500 |
} db2
|
sl@0
|
501 |
} {1 2 3 4}
|
sl@0
|
502 |
}
|
sl@0
|
503 |
do_test shared-$av.6.1.3 {
|
sl@0
|
504 |
# Establish a write lock on table t2 via connection db2. Then make a
|
sl@0
|
505 |
# UNION all query using connection db1 that first accesses t1, followed
|
sl@0
|
506 |
# by t2. If the locks are grabbed at the start of the statement (as
|
sl@0
|
507 |
# they should be), no rows are returned. If (as was previously the case)
|
sl@0
|
508 |
# they are grabbed as the tables are accessed, the t1 rows will be
|
sl@0
|
509 |
# returned before the query fails.
|
sl@0
|
510 |
#
|
sl@0
|
511 |
execsql {
|
sl@0
|
512 |
BEGIN;
|
sl@0
|
513 |
INSERT INTO t2 VALUES(5, 6);
|
sl@0
|
514 |
} db2
|
sl@0
|
515 |
set ret [list]
|
sl@0
|
516 |
catch {
|
sl@0
|
517 |
db1 eval {SELECT * FROM t1 UNION ALL SELECT * FROM t2} {
|
sl@0
|
518 |
lappend ret $a $b
|
sl@0
|
519 |
}
|
sl@0
|
520 |
}
|
sl@0
|
521 |
set ret
|
sl@0
|
522 |
} {}
|
sl@0
|
523 |
do_test shared-$av.6.1.4 {
|
sl@0
|
524 |
execsql {
|
sl@0
|
525 |
COMMIT;
|
sl@0
|
526 |
BEGIN;
|
sl@0
|
527 |
INSERT INTO t1 VALUES(7, 8);
|
sl@0
|
528 |
} db2
|
sl@0
|
529 |
set ret [list]
|
sl@0
|
530 |
catch {
|
sl@0
|
531 |
db1 eval {
|
sl@0
|
532 |
SELECT (CASE WHEN a>4 THEN (SELECT a FROM t1) ELSE 0 END) AS d FROM t2;
|
sl@0
|
533 |
} {
|
sl@0
|
534 |
lappend ret $d
|
sl@0
|
535 |
}
|
sl@0
|
536 |
}
|
sl@0
|
537 |
set ret
|
sl@0
|
538 |
} {}
|
sl@0
|
539 |
|
sl@0
|
540 |
catch {db1 close}
|
sl@0
|
541 |
catch {db2 close}
|
sl@0
|
542 |
foreach f [list test.db test2.db] {
|
sl@0
|
543 |
file delete -force $f ${f}-journal
|
sl@0
|
544 |
}
|
sl@0
|
545 |
|
sl@0
|
546 |
#--------------------------------------------------------------------------
|
sl@0
|
547 |
# Tests shared-7.* test auto-vacuum does not invalidate cursors from
|
sl@0
|
548 |
# other shared-cache users when it reorganizes the database on
|
sl@0
|
549 |
# COMMIT.
|
sl@0
|
550 |
#
|
sl@0
|
551 |
do_test shared-$av.7.1 {
|
sl@0
|
552 |
# This test case sets up a test database in auto-vacuum mode consisting
|
sl@0
|
553 |
# of two tables, t1 and t2. Both have a single index. Table t1 is
|
sl@0
|
554 |
# populated first (so consists of pages toward the start of the db file),
|
sl@0
|
555 |
# t2 second (pages toward the end of the file).
|
sl@0
|
556 |
sqlite3 db test.db
|
sl@0
|
557 |
sqlite3 db2 test.db
|
sl@0
|
558 |
execsql {
|
sl@0
|
559 |
BEGIN;
|
sl@0
|
560 |
CREATE TABLE t1(a PRIMARY KEY, b);
|
sl@0
|
561 |
CREATE TABLE t2(a PRIMARY KEY, b);
|
sl@0
|
562 |
}
|
sl@0
|
563 |
set ::contents {}
|
sl@0
|
564 |
for {set i 0} {$i < 100} {incr i} {
|
sl@0
|
565 |
set a [string repeat "$i " 20]
|
sl@0
|
566 |
set b [string repeat "$i " 20]
|
sl@0
|
567 |
db eval {
|
sl@0
|
568 |
INSERT INTO t1 VALUES(:a, :b);
|
sl@0
|
569 |
}
|
sl@0
|
570 |
lappend ::contents [list [expr $i+1] $a $b]
|
sl@0
|
571 |
}
|
sl@0
|
572 |
execsql {
|
sl@0
|
573 |
INSERT INTO t2 SELECT * FROM t1;
|
sl@0
|
574 |
COMMIT;
|
sl@0
|
575 |
}
|
sl@0
|
576 |
} {}
|
sl@0
|
577 |
|
sl@0
|
578 |
#
|
sl@0
|
579 |
# Symbian: "stack overflow" if "shared-$av.7.2" is executed
|
sl@0
|
580 |
#
|
sl@0
|
581 |
if {$tcl_platform(platform)!="symbian"} {
|
sl@0
|
582 |
do_test shared-$av.7.2 {
|
sl@0
|
583 |
# This test case deletes the contents of table t1 (the one at the start of
|
sl@0
|
584 |
# the file) while many cursors are open on table t2 and its index. All of
|
sl@0
|
585 |
# the non-root pages will be moved from the end to the start of the file
|
sl@0
|
586 |
# when the DELETE is committed - this test verifies that moving the pages
|
sl@0
|
587 |
# does not disturb the open cursors.
|
sl@0
|
588 |
#
|
sl@0
|
589 |
|
sl@0
|
590 |
proc lockrow {db tbl oids body} {
|
sl@0
|
591 |
set ret [list]
|
sl@0
|
592 |
db eval "SELECT oid AS i, a, b FROM $tbl ORDER BY a" {
|
sl@0
|
593 |
if {$i==[lindex $oids 0]} {
|
sl@0
|
594 |
set noids [lrange $oids 1 end]
|
sl@0
|
595 |
if {[llength $noids]==0} {
|
sl@0
|
596 |
set subret [eval $body]
|
sl@0
|
597 |
} else {
|
sl@0
|
598 |
set subret [lockrow $db $tbl $noids $body]
|
sl@0
|
599 |
}
|
sl@0
|
600 |
}
|
sl@0
|
601 |
lappend ret [list $i $a $b]
|
sl@0
|
602 |
}
|
sl@0
|
603 |
return [linsert $subret 0 $ret]
|
sl@0
|
604 |
}
|
sl@0
|
605 |
proc locktblrows {db tbl body} {
|
sl@0
|
606 |
set oids [db eval "SELECT oid FROM $tbl"]
|
sl@0
|
607 |
lockrow $db $tbl $oids $body
|
sl@0
|
608 |
}
|
sl@0
|
609 |
|
sl@0
|
610 |
set scans [locktblrows db t2 {
|
sl@0
|
611 |
execsql {
|
sl@0
|
612 |
DELETE FROM t1;
|
sl@0
|
613 |
} db2
|
sl@0
|
614 |
}]
|
sl@0
|
615 |
set error 0
|
sl@0
|
616 |
|
sl@0
|
617 |
# Test that each SELECT query returned the expected contents of t2.
|
sl@0
|
618 |
foreach s $scans {
|
sl@0
|
619 |
if {[lsort -integer -index 0 $s]!=$::contents} {
|
sl@0
|
620 |
set error 1
|
sl@0
|
621 |
}
|
sl@0
|
622 |
}
|
sl@0
|
623 |
set error
|
sl@0
|
624 |
} {0}
|
sl@0
|
625 |
}
|
sl@0
|
626 |
|
sl@0
|
627 |
catch {db close}
|
sl@0
|
628 |
catch {db2 close}
|
sl@0
|
629 |
unset -nocomplain contents
|
sl@0
|
630 |
|
sl@0
|
631 |
#--------------------------------------------------------------------------
|
sl@0
|
632 |
# The following tests try to trick the shared-cache code into assuming
|
sl@0
|
633 |
# the wrong encoding for a database.
|
sl@0
|
634 |
#
|
sl@0
|
635 |
file delete -force test.db test.db-journal
|
sl@0
|
636 |
ifcapable utf16 {
|
sl@0
|
637 |
do_test shared-$av.8.1.1 {
|
sl@0
|
638 |
sqlite3 db test.db
|
sl@0
|
639 |
execsql {
|
sl@0
|
640 |
PRAGMA encoding = 'UTF-16';
|
sl@0
|
641 |
SELECT * FROM sqlite_master;
|
sl@0
|
642 |
}
|
sl@0
|
643 |
} {}
|
sl@0
|
644 |
do_test shared-$av.8.1.2 {
|
sl@0
|
645 |
string range [execsql {PRAGMA encoding;}] 0 end-2
|
sl@0
|
646 |
} {UTF-16}
|
sl@0
|
647 |
|
sl@0
|
648 |
do_test shared-$av.8.1.3 {
|
sl@0
|
649 |
sqlite3 db2 test.db
|
sl@0
|
650 |
execsql {
|
sl@0
|
651 |
PRAGMA encoding = 'UTF-8';
|
sl@0
|
652 |
CREATE TABLE abc(a, b, c);
|
sl@0
|
653 |
} db2
|
sl@0
|
654 |
} {}
|
sl@0
|
655 |
do_test shared-$av.8.1.4 {
|
sl@0
|
656 |
execsql {
|
sl@0
|
657 |
SELECT * FROM sqlite_master;
|
sl@0
|
658 |
}
|
sl@0
|
659 |
} "table abc abc [expr $AUTOVACUUM?3:2] {CREATE TABLE abc(a, b, c)}"
|
sl@0
|
660 |
do_test shared-$av.8.1.5 {
|
sl@0
|
661 |
db2 close
|
sl@0
|
662 |
execsql {
|
sl@0
|
663 |
PRAGMA encoding;
|
sl@0
|
664 |
}
|
sl@0
|
665 |
} {UTF-8}
|
sl@0
|
666 |
|
sl@0
|
667 |
file delete -force test2.db test2.db-journal
|
sl@0
|
668 |
do_test shared-$av.8.2.1 {
|
sl@0
|
669 |
execsql {
|
sl@0
|
670 |
ATTACH 'test2.db' AS aux;
|
sl@0
|
671 |
SELECT * FROM aux.sqlite_master;
|
sl@0
|
672 |
}
|
sl@0
|
673 |
} {}
|
sl@0
|
674 |
do_test shared-$av.8.2.2 {
|
sl@0
|
675 |
sqlite3 db2 test2.db
|
sl@0
|
676 |
execsql {
|
sl@0
|
677 |
PRAGMA encoding = 'UTF-16';
|
sl@0
|
678 |
CREATE TABLE def(d, e, f);
|
sl@0
|
679 |
} db2
|
sl@0
|
680 |
string range [execsql {PRAGMA encoding;} db2] 0 end-2
|
sl@0
|
681 |
} {UTF-16}
|
sl@0
|
682 |
|
sl@0
|
683 |
catch {db close}
|
sl@0
|
684 |
catch {db2 close}
|
sl@0
|
685 |
file delete -force test.db test2.db
|
sl@0
|
686 |
|
sl@0
|
687 |
do_test shared-$av.8.3.2 {
|
sl@0
|
688 |
sqlite3 db test.db
|
sl@0
|
689 |
execsql { CREATE TABLE def(d, e, f) }
|
sl@0
|
690 |
execsql { PRAGMA encoding }
|
sl@0
|
691 |
} {UTF-8}
|
sl@0
|
692 |
do_test shared-$av.8.3.3 {
|
sl@0
|
693 |
set zDb16 "[encoding convertto unicode test.db]\x00\x00"
|
sl@0
|
694 |
set db16 [sqlite3_open16 $zDb16 {}]
|
sl@0
|
695 |
|
sl@0
|
696 |
set stmt [sqlite3_prepare $db16 "SELECT sql FROM sqlite_master" -1 DUMMY]
|
sl@0
|
697 |
sqlite3_step $stmt
|
sl@0
|
698 |
set sql [sqlite3_column_text $stmt 0]
|
sl@0
|
699 |
sqlite3_finalize $stmt
|
sl@0
|
700 |
set sql
|
sl@0
|
701 |
} {CREATE TABLE def(d, e, f)}
|
sl@0
|
702 |
do_test shared-$av.8.3.4 {
|
sl@0
|
703 |
set stmt [sqlite3_prepare $db16 "PRAGMA encoding" -1 DUMMY]
|
sl@0
|
704 |
sqlite3_step $stmt
|
sl@0
|
705 |
set enc [sqlite3_column_text $stmt 0]
|
sl@0
|
706 |
sqlite3_finalize $stmt
|
sl@0
|
707 |
set enc
|
sl@0
|
708 |
} {UTF-8}
|
sl@0
|
709 |
|
sl@0
|
710 |
sqlite3_close $db16
|
sl@0
|
711 |
|
sl@0
|
712 |
# Bug #2547 is causing this to fail.
|
sl@0
|
713 |
if 0 {
|
sl@0
|
714 |
do_test shared-$av.8.2.3 {
|
sl@0
|
715 |
catchsql {
|
sl@0
|
716 |
SELECT * FROM aux.sqlite_master;
|
sl@0
|
717 |
}
|
sl@0
|
718 |
} {1 {attached databases must use the same text encoding as main database}}
|
sl@0
|
719 |
}
|
sl@0
|
720 |
}
|
sl@0
|
721 |
|
sl@0
|
722 |
catch {db close}
|
sl@0
|
723 |
catch {db2 close}
|
sl@0
|
724 |
file delete -force test.db test2.db
|
sl@0
|
725 |
|
sl@0
|
726 |
#---------------------------------------------------------------------------
|
sl@0
|
727 |
# The following tests - shared-9.* - test interactions between TEMP triggers
|
sl@0
|
728 |
# and shared-schemas.
|
sl@0
|
729 |
#
|
sl@0
|
730 |
ifcapable trigger&&tempdb {
|
sl@0
|
731 |
|
sl@0
|
732 |
do_test shared-$av.9.1 {
|
sl@0
|
733 |
sqlite3 db test.db
|
sl@0
|
734 |
sqlite3 db2 test.db
|
sl@0
|
735 |
execsql {
|
sl@0
|
736 |
CREATE TABLE abc(a, b, c);
|
sl@0
|
737 |
CREATE TABLE abc_mirror(a, b, c);
|
sl@0
|
738 |
CREATE TEMP TRIGGER BEFORE INSERT ON abc BEGIN
|
sl@0
|
739 |
INSERT INTO abc_mirror(a, b, c) VALUES(new.a, new.b, new.c);
|
sl@0
|
740 |
END;
|
sl@0
|
741 |
INSERT INTO abc VALUES(1, 2, 3);
|
sl@0
|
742 |
SELECT * FROM abc_mirror;
|
sl@0
|
743 |
}
|
sl@0
|
744 |
} {1 2 3}
|
sl@0
|
745 |
do_test shared-$av.9.2 {
|
sl@0
|
746 |
execsql {
|
sl@0
|
747 |
INSERT INTO abc VALUES(4, 5, 6);
|
sl@0
|
748 |
SELECT * FROM abc_mirror;
|
sl@0
|
749 |
} db2
|
sl@0
|
750 |
} {1 2 3}
|
sl@0
|
751 |
do_test shared-$av.9.3 {
|
sl@0
|
752 |
db close
|
sl@0
|
753 |
db2 close
|
sl@0
|
754 |
} {}
|
sl@0
|
755 |
|
sl@0
|
756 |
} ; # End shared-9.*
|
sl@0
|
757 |
|
sl@0
|
758 |
#---------------------------------------------------------------------------
|
sl@0
|
759 |
# The following tests - shared-10.* - test that the library behaves
|
sl@0
|
760 |
# correctly when a connection to a shared-cache is closed.
|
sl@0
|
761 |
#
|
sl@0
|
762 |
do_test shared-$av.10.1 {
|
sl@0
|
763 |
# Create a small sample database with two connections to it (db and db2).
|
sl@0
|
764 |
file delete -force test.db
|
sl@0
|
765 |
sqlite3 db test.db
|
sl@0
|
766 |
sqlite3 db2 test.db
|
sl@0
|
767 |
execsql {
|
sl@0
|
768 |
CREATE TABLE ab(a PRIMARY KEY, b);
|
sl@0
|
769 |
CREATE TABLE de(d PRIMARY KEY, e);
|
sl@0
|
770 |
INSERT INTO ab VALUES('Chiang Mai', 100000);
|
sl@0
|
771 |
INSERT INTO ab VALUES('Bangkok', 8000000);
|
sl@0
|
772 |
INSERT INTO de VALUES('Ubon', 120000);
|
sl@0
|
773 |
INSERT INTO de VALUES('Khon Kaen', 200000);
|
sl@0
|
774 |
}
|
sl@0
|
775 |
} {}
|
sl@0
|
776 |
do_test shared-$av.10.2 {
|
sl@0
|
777 |
# Open a read-transaction with the first connection, a write-transaction
|
sl@0
|
778 |
# with the second.
|
sl@0
|
779 |
execsql {
|
sl@0
|
780 |
BEGIN;
|
sl@0
|
781 |
SELECT * FROM ab;
|
sl@0
|
782 |
}
|
sl@0
|
783 |
execsql {
|
sl@0
|
784 |
BEGIN;
|
sl@0
|
785 |
INSERT INTO de VALUES('Pataya', 30000);
|
sl@0
|
786 |
} db2
|
sl@0
|
787 |
} {}
|
sl@0
|
788 |
do_test shared-$av.10.3 {
|
sl@0
|
789 |
# An external connection should be able to read the database, but not
|
sl@0
|
790 |
# prepare a write operation.
|
sl@0
|
791 |
if {$::tcl_platform(platform)=="unix"} {
|
sl@0
|
792 |
sqlite3 db3 ./test.db
|
sl@0
|
793 |
} else {
|
sl@0
|
794 |
sqlite3 db3 TEST.DB
|
sl@0
|
795 |
}
|
sl@0
|
796 |
execsql {
|
sl@0
|
797 |
SELECT * FROM ab;
|
sl@0
|
798 |
} db3
|
sl@0
|
799 |
catchsql {
|
sl@0
|
800 |
BEGIN;
|
sl@0
|
801 |
INSERT INTO de VALUES('Pataya', 30000);
|
sl@0
|
802 |
} db3
|
sl@0
|
803 |
} {1 {database is locked}}
|
sl@0
|
804 |
do_test shared-$av.10.4 {
|
sl@0
|
805 |
# Close the connection with the write-transaction open
|
sl@0
|
806 |
db2 close
|
sl@0
|
807 |
} {}
|
sl@0
|
808 |
do_test shared-$av.10.5 {
|
sl@0
|
809 |
# Test that the db2 transaction has been automatically rolled back.
|
sl@0
|
810 |
# If it has not the ('Pataya', 30000) entry will still be in the table.
|
sl@0
|
811 |
execsql {
|
sl@0
|
812 |
SELECT * FROM de;
|
sl@0
|
813 |
}
|
sl@0
|
814 |
} {Ubon 120000 {Khon Kaen} 200000}
|
sl@0
|
815 |
do_test shared-$av.10.5 {
|
sl@0
|
816 |
# Closing db2 should have dropped the shared-cache back to a read-lock.
|
sl@0
|
817 |
# So db3 should be able to prepare a write...
|
sl@0
|
818 |
catchsql {INSERT INTO de VALUES('Pataya', 30000);} db3
|
sl@0
|
819 |
} {0 {}}
|
sl@0
|
820 |
do_test shared-$av.10.6 {
|
sl@0
|
821 |
# ... but not commit it.
|
sl@0
|
822 |
catchsql {COMMIT} db3
|
sl@0
|
823 |
} {1 {database is locked}}
|
sl@0
|
824 |
do_test shared-$av.10.7 {
|
sl@0
|
825 |
# Commit the (read-only) db transaction. Check via db3 to make sure the
|
sl@0
|
826 |
# contents of table "de" are still as they should be.
|
sl@0
|
827 |
execsql {
|
sl@0
|
828 |
COMMIT;
|
sl@0
|
829 |
}
|
sl@0
|
830 |
execsql {
|
sl@0
|
831 |
SELECT * FROM de;
|
sl@0
|
832 |
} db3
|
sl@0
|
833 |
} {Ubon 120000 {Khon Kaen} 200000 Pataya 30000}
|
sl@0
|
834 |
do_test shared-$av.10.9 {
|
sl@0
|
835 |
# Commit the external transaction.
|
sl@0
|
836 |
catchsql {COMMIT} db3
|
sl@0
|
837 |
} {0 {}}
|
sl@0
|
838 |
integrity_check shared-$av.10.10
|
sl@0
|
839 |
do_test shared-$av.10.11 {
|
sl@0
|
840 |
db close
|
sl@0
|
841 |
db3 close
|
sl@0
|
842 |
} {}
|
sl@0
|
843 |
|
sl@0
|
844 |
do_test shared-$av.11.1 {
|
sl@0
|
845 |
file delete -force test.db
|
sl@0
|
846 |
sqlite3 db test.db
|
sl@0
|
847 |
sqlite3 db2 test.db
|
sl@0
|
848 |
execsql {
|
sl@0
|
849 |
CREATE TABLE abc(a, b, c);
|
sl@0
|
850 |
CREATE TABLE abc2(a, b, c);
|
sl@0
|
851 |
BEGIN;
|
sl@0
|
852 |
INSERT INTO abc VALUES(1, 2, 3);
|
sl@0
|
853 |
}
|
sl@0
|
854 |
} {}
|
sl@0
|
855 |
do_test shared-$av.11.2 {
|
sl@0
|
856 |
catchsql {BEGIN;} db2
|
sl@0
|
857 |
catchsql {SELECT * FROM abc;} db2
|
sl@0
|
858 |
} {1 {database table is locked: abc}}
|
sl@0
|
859 |
do_test shared-$av.11.3 {
|
sl@0
|
860 |
catchsql {BEGIN} db2
|
sl@0
|
861 |
} {1 {cannot start a transaction within a transaction}}
|
sl@0
|
862 |
do_test shared-$av.11.4 {
|
sl@0
|
863 |
catchsql {SELECT * FROM abc2;} db2
|
sl@0
|
864 |
} {0 {}}
|
sl@0
|
865 |
do_test shared-$av.11.5 {
|
sl@0
|
866 |
catchsql {INSERT INTO abc2 VALUES(1, 2, 3);} db2
|
sl@0
|
867 |
} {1 {database is locked}}
|
sl@0
|
868 |
do_test shared-$av.11.6 {
|
sl@0
|
869 |
catchsql {SELECT * FROM abc2}
|
sl@0
|
870 |
} {0 {}}
|
sl@0
|
871 |
do_test shared-$av.11.6 {
|
sl@0
|
872 |
execsql {
|
sl@0
|
873 |
ROLLBACK;
|
sl@0
|
874 |
PRAGMA read_uncommitted = 1;
|
sl@0
|
875 |
} db2
|
sl@0
|
876 |
} {}
|
sl@0
|
877 |
do_test shared-$av.11.7 {
|
sl@0
|
878 |
execsql {
|
sl@0
|
879 |
INSERT INTO abc2 VALUES(4, 5, 6);
|
sl@0
|
880 |
INSERT INTO abc2 VALUES(7, 8, 9);
|
sl@0
|
881 |
}
|
sl@0
|
882 |
} {}
|
sl@0
|
883 |
do_test shared-$av.11.8 {
|
sl@0
|
884 |
set res [list]
|
sl@0
|
885 |
db2 eval {
|
sl@0
|
886 |
SELECT abc.a as I, abc2.a as II FROM abc, abc2;
|
sl@0
|
887 |
} {
|
sl@0
|
888 |
execsql {
|
sl@0
|
889 |
DELETE FROM abc WHERE 1;
|
sl@0
|
890 |
}
|
sl@0
|
891 |
lappend res $I $II
|
sl@0
|
892 |
}
|
sl@0
|
893 |
set res
|
sl@0
|
894 |
} {1 4 {} 7}
|
sl@0
|
895 |
if {[llength [info command sqlite3_shared_cache_report]]==1} {
|
sl@0
|
896 |
do_test shared-$av.11.9 {
|
sl@0
|
897 |
string tolower [sqlite3_shared_cache_report]
|
sl@0
|
898 |
} [string tolower [list [file nativename [file normalize test.db]] 2]]
|
sl@0
|
899 |
}
|
sl@0
|
900 |
|
sl@0
|
901 |
do_test shared-$av.11.11 {
|
sl@0
|
902 |
db close
|
sl@0
|
903 |
db2 close
|
sl@0
|
904 |
} {}
|
sl@0
|
905 |
|
sl@0
|
906 |
# This tests that if it is impossible to free any pages, SQLite will
|
sl@0
|
907 |
# exceed the limit set by PRAGMA cache_size.
|
sl@0
|
908 |
file delete -force test.db test.db-journal
|
sl@0
|
909 |
sqlite3 db test.db
|
sl@0
|
910 |
ifcapable pager_pragmas {
|
sl@0
|
911 |
do_test shared-$av.12.1 {
|
sl@0
|
912 |
execsql {
|
sl@0
|
913 |
PRAGMA cache_size = 10;
|
sl@0
|
914 |
PRAGMA cache_size;
|
sl@0
|
915 |
}
|
sl@0
|
916 |
} {10}
|
sl@0
|
917 |
}
|
sl@0
|
918 |
do_test shared-$av.12.2 {
|
sl@0
|
919 |
set ::db_handles [list]
|
sl@0
|
920 |
for {set i 1} {$i < 15} {incr i} {
|
sl@0
|
921 |
lappend ::db_handles db$i
|
sl@0
|
922 |
sqlite3 db$i test.db
|
sl@0
|
923 |
execsql "CREATE TABLE db${i}(a, b, c)" db$i
|
sl@0
|
924 |
execsql "INSERT INTO db${i} VALUES(1, 2, 3)"
|
sl@0
|
925 |
}
|
sl@0
|
926 |
} {}
|
sl@0
|
927 |
proc nested_select {handles} {
|
sl@0
|
928 |
[lindex $handles 0] eval "SELECT * FROM [lindex $handles 0]" {
|
sl@0
|
929 |
lappend ::res $a $b $c
|
sl@0
|
930 |
if {[llength $handles]>1} {
|
sl@0
|
931 |
nested_select [lrange $handles 1 end]
|
sl@0
|
932 |
}
|
sl@0
|
933 |
}
|
sl@0
|
934 |
}
|
sl@0
|
935 |
do_test shared-$av.12.3 {
|
sl@0
|
936 |
set ::res [list]
|
sl@0
|
937 |
nested_select $::db_handles
|
sl@0
|
938 |
set ::res
|
sl@0
|
939 |
} [string range [string repeat "1 2 3 " [llength $::db_handles]] 0 end-1]
|
sl@0
|
940 |
|
sl@0
|
941 |
do_test shared-$av.12.X {
|
sl@0
|
942 |
db close
|
sl@0
|
943 |
foreach h $::db_handles {
|
sl@0
|
944 |
$h close
|
sl@0
|
945 |
}
|
sl@0
|
946 |
} {}
|
sl@0
|
947 |
|
sl@0
|
948 |
# Internally, locks are acquired on shared B-Tree structures in the order
|
sl@0
|
949 |
# that the structures appear in the virtual memory address space. This
|
sl@0
|
950 |
# test case attempts to cause the order of the structures in memory
|
sl@0
|
951 |
# to be different from the order in which they are attached to a given
|
sl@0
|
952 |
# database handle. This covers an extra line or two.
|
sl@0
|
953 |
#
|
sl@0
|
954 |
do_test shared-$av.13.1 {
|
sl@0
|
955 |
file delete -force test2.db test3.db test4.db test5.db
|
sl@0
|
956 |
sqlite3 db :memory:
|
sl@0
|
957 |
execsql {
|
sl@0
|
958 |
ATTACH 'test2.db' AS aux2;
|
sl@0
|
959 |
ATTACH 'test3.db' AS aux3;
|
sl@0
|
960 |
ATTACH 'test4.db' AS aux4;
|
sl@0
|
961 |
ATTACH 'test5.db' AS aux5;
|
sl@0
|
962 |
DETACH aux2;
|
sl@0
|
963 |
DETACH aux3;
|
sl@0
|
964 |
DETACH aux4;
|
sl@0
|
965 |
ATTACH 'test2.db' AS aux2;
|
sl@0
|
966 |
ATTACH 'test3.db' AS aux3;
|
sl@0
|
967 |
ATTACH 'test4.db' AS aux4;
|
sl@0
|
968 |
}
|
sl@0
|
969 |
} {}
|
sl@0
|
970 |
do_test shared-$av.13.2 {
|
sl@0
|
971 |
execsql {
|
sl@0
|
972 |
CREATE TABLE t1(a, b, c);
|
sl@0
|
973 |
CREATE TABLE aux2.t2(a, b, c);
|
sl@0
|
974 |
CREATE TABLE aux3.t3(a, b, c);
|
sl@0
|
975 |
CREATE TABLE aux4.t4(a, b, c);
|
sl@0
|
976 |
CREATE TABLE aux5.t5(a, b, c);
|
sl@0
|
977 |
SELECT count(*) FROM
|
sl@0
|
978 |
aux2.sqlite_master,
|
sl@0
|
979 |
aux3.sqlite_master,
|
sl@0
|
980 |
aux4.sqlite_master,
|
sl@0
|
981 |
aux5.sqlite_master
|
sl@0
|
982 |
}
|
sl@0
|
983 |
} {1}
|
sl@0
|
984 |
do_test shared-$av.13.3 {
|
sl@0
|
985 |
db close
|
sl@0
|
986 |
} {}
|
sl@0
|
987 |
|
sl@0
|
988 |
# Test that nothing horrible happens if a connection to a shared B-Tree
|
sl@0
|
989 |
# structure is closed while some other connection has an open cursor.
|
sl@0
|
990 |
#
|
sl@0
|
991 |
do_test shared-$av.14.1 {
|
sl@0
|
992 |
sqlite3 db test.db
|
sl@0
|
993 |
sqlite3 db2 test.db
|
sl@0
|
994 |
execsql {SELECT name FROM sqlite_master}
|
sl@0
|
995 |
} {db1 db2 db3 db4 db5 db6 db7 db8 db9 db10 db11 db12 db13 db14}
|
sl@0
|
996 |
do_test shared-$av.14.2 {
|
sl@0
|
997 |
set res [list]
|
sl@0
|
998 |
db eval {SELECT name FROM sqlite_master} {
|
sl@0
|
999 |
if {$name eq "db7"} {
|
sl@0
|
1000 |
db2 close
|
sl@0
|
1001 |
}
|
sl@0
|
1002 |
lappend res $name
|
sl@0
|
1003 |
}
|
sl@0
|
1004 |
set res
|
sl@0
|
1005 |
} {db1 db2 db3 db4 db5 db6 db7 db8 db9 db10 db11 db12 db13 db14}
|
sl@0
|
1006 |
do_test shared-$av.14.3 {
|
sl@0
|
1007 |
db close
|
sl@0
|
1008 |
} {}
|
sl@0
|
1009 |
|
sl@0
|
1010 |
}
|
sl@0
|
1011 |
|
sl@0
|
1012 |
sqlite3_enable_shared_cache $::enable_shared_cache
|
sl@0
|
1013 |
finish_test
|