sl@0
|
1 |
# 2006 July 25
|
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 focus
|
sl@0
|
12 |
# of this test is reading and writing to the database from within a
|
sl@0
|
13 |
# virtual table xSync() callback.
|
sl@0
|
14 |
#
|
sl@0
|
15 |
# $Id: vtab7.test,v 1.4 2007/12/04 16:54:53 drh Exp $
|
sl@0
|
16 |
|
sl@0
|
17 |
set testdir [file dirname $argv0]
|
sl@0
|
18 |
source $testdir/tester.tcl
|
sl@0
|
19 |
|
sl@0
|
20 |
ifcapable !vtab {
|
sl@0
|
21 |
finish_test
|
sl@0
|
22 |
return
|
sl@0
|
23 |
}
|
sl@0
|
24 |
|
sl@0
|
25 |
# Register the echo module. Code inside the echo module appends elements
|
sl@0
|
26 |
# to the global tcl list variable ::echo_module whenever SQLite invokes
|
sl@0
|
27 |
# certain module callbacks. This includes the xSync(), xCommit() and
|
sl@0
|
28 |
# xRollback() callbacks. For each of these callback, two elements are
|
sl@0
|
29 |
# appended to ::echo_module, as follows:
|
sl@0
|
30 |
#
|
sl@0
|
31 |
# Module method Elements appended to ::echo_module
|
sl@0
|
32 |
# -------------------------------------------------------
|
sl@0
|
33 |
# xSync() xSync echo($tablename)
|
sl@0
|
34 |
# xCommit() xCommit echo($tablename)
|
sl@0
|
35 |
# xRollback() xRollback echo($tablename)
|
sl@0
|
36 |
# -------------------------------------------------------
|
sl@0
|
37 |
#
|
sl@0
|
38 |
# In each case, $tablename is replaced by the name of the real table (not
|
sl@0
|
39 |
# the echo table). By setting up a tcl trace on the ::echo_module variable,
|
sl@0
|
40 |
# code in this file arranges for a Tcl script to be executed from within
|
sl@0
|
41 |
# the echo module xSync() callback.
|
sl@0
|
42 |
#
|
sl@0
|
43 |
register_echo_module [sqlite3_connection_pointer db]
|
sl@0
|
44 |
trace add variable ::echo_module write echo_module_trace
|
sl@0
|
45 |
|
sl@0
|
46 |
# This Tcl proc is invoked whenever the ::echo_module variable is written.
|
sl@0
|
47 |
#
|
sl@0
|
48 |
proc echo_module_trace {args} {
|
sl@0
|
49 |
# Filter out writes to ::echo_module that are not xSync, xCommit or
|
sl@0
|
50 |
# xRollback callbacks.
|
sl@0
|
51 |
if {[llength $::echo_module] < 2} return
|
sl@0
|
52 |
set x [lindex $::echo_module end-1]
|
sl@0
|
53 |
if {$x ne "xSync" && $x ne "xCommit" && $x ne "xRollback"} return
|
sl@0
|
54 |
|
sl@0
|
55 |
regexp {^echo.(.*).$} [lindex $::echo_module end] dummy tablename
|
sl@0
|
56 |
# puts "Ladies and gentlemen, an $x on $tablename!"
|
sl@0
|
57 |
|
sl@0
|
58 |
if {[info exists ::callbacks($x,$tablename)]} {
|
sl@0
|
59 |
eval $::callbacks($x,$tablename)
|
sl@0
|
60 |
}
|
sl@0
|
61 |
}
|
sl@0
|
62 |
|
sl@0
|
63 |
# The following tests, vtab7-1.*, test that the trace callback on
|
sl@0
|
64 |
# ::echo_module is providing the expected tcl callbacks.
|
sl@0
|
65 |
do_test vtab7-1.1 {
|
sl@0
|
66 |
execsql {
|
sl@0
|
67 |
CREATE TABLE abc(a, b, c);
|
sl@0
|
68 |
CREATE VIRTUAL TABLE abc2 USING echo(abc);
|
sl@0
|
69 |
}
|
sl@0
|
70 |
} {}
|
sl@0
|
71 |
|
sl@0
|
72 |
do_test vtab7-1.2 {
|
sl@0
|
73 |
set ::callbacks(xSync,abc) {incr ::counter}
|
sl@0
|
74 |
set ::counter 0
|
sl@0
|
75 |
execsql {
|
sl@0
|
76 |
INSERT INTO abc2 VALUES(1, 2, 3);
|
sl@0
|
77 |
}
|
sl@0
|
78 |
set ::counter
|
sl@0
|
79 |
} {1}
|
sl@0
|
80 |
|
sl@0
|
81 |
# Write to an existing database table from within an xSync callback.
|
sl@0
|
82 |
do_test vtab7-2.1 {
|
sl@0
|
83 |
set ::callbacks(xSync,abc) {
|
sl@0
|
84 |
execsql {INSERT INTO log VALUES('xSync');}
|
sl@0
|
85 |
}
|
sl@0
|
86 |
execsql {
|
sl@0
|
87 |
CREATE TABLE log(msg);
|
sl@0
|
88 |
INSERT INTO abc2 VALUES(4, 5, 6);
|
sl@0
|
89 |
SELECT * FROM log;
|
sl@0
|
90 |
}
|
sl@0
|
91 |
} {xSync}
|
sl@0
|
92 |
do_test vtab7-2.3 {
|
sl@0
|
93 |
execsql {
|
sl@0
|
94 |
INSERT INTO abc2 VALUES(4, 5, 6);
|
sl@0
|
95 |
SELECT * FROM log;
|
sl@0
|
96 |
}
|
sl@0
|
97 |
} {xSync xSync}
|
sl@0
|
98 |
do_test vtab7-2.4 {
|
sl@0
|
99 |
execsql {
|
sl@0
|
100 |
INSERT INTO abc2 VALUES(4, 5, 6);
|
sl@0
|
101 |
SELECT * FROM log;
|
sl@0
|
102 |
}
|
sl@0
|
103 |
} {xSync xSync xSync}
|
sl@0
|
104 |
|
sl@0
|
105 |
# Create a database table from within xSync callback.
|
sl@0
|
106 |
do_test vtab7-2.5 {
|
sl@0
|
107 |
set ::callbacks(xSync,abc) {
|
sl@0
|
108 |
execsql { CREATE TABLE newtab(d, e, f); }
|
sl@0
|
109 |
}
|
sl@0
|
110 |
execsql {
|
sl@0
|
111 |
INSERT INTO abc2 VALUES(1, 2, 3);
|
sl@0
|
112 |
SELECT name FROM sqlite_master ORDER BY name;
|
sl@0
|
113 |
}
|
sl@0
|
114 |
} {abc abc2 log newtab}
|
sl@0
|
115 |
|
sl@0
|
116 |
# Drop a database table from within xSync callback.
|
sl@0
|
117 |
# This is not allowed. Tables cannot be dropped while
|
sl@0
|
118 |
# any other statement is active.
|
sl@0
|
119 |
#
|
sl@0
|
120 |
do_test vtab7-2.6 {
|
sl@0
|
121 |
set ::callbacks(xSync,abc) {
|
sl@0
|
122 |
set ::rc [catchsql { DROP TABLE newtab }]
|
sl@0
|
123 |
}
|
sl@0
|
124 |
execsql {
|
sl@0
|
125 |
INSERT INTO abc2 VALUES(1, 2, 3);
|
sl@0
|
126 |
SELECT name FROM sqlite_master ORDER BY name;
|
sl@0
|
127 |
}
|
sl@0
|
128 |
} {abc abc2 log newtab}
|
sl@0
|
129 |
do_test vtab7-2.6.1 {
|
sl@0
|
130 |
set ::rc
|
sl@0
|
131 |
} {1 {database table is locked}}
|
sl@0
|
132 |
execsql {DROP TABLE newtab}
|
sl@0
|
133 |
|
sl@0
|
134 |
# Write to an attached database from xSync().
|
sl@0
|
135 |
ifcapable attach {
|
sl@0
|
136 |
do_test vtab7-3.1 {
|
sl@0
|
137 |
file delete -force test2.db
|
sl@0
|
138 |
file delete -force test2.db-journal
|
sl@0
|
139 |
execsql {
|
sl@0
|
140 |
ATTACH 'test2.db' AS db2;
|
sl@0
|
141 |
CREATE TABLE db2.stuff(description, shape, color);
|
sl@0
|
142 |
}
|
sl@0
|
143 |
set ::callbacks(xSync,abc) {
|
sl@0
|
144 |
execsql { INSERT INTO db2.stuff VALUES('abc', 'square', 'green'); }
|
sl@0
|
145 |
}
|
sl@0
|
146 |
execsql {
|
sl@0
|
147 |
INSERT INTO abc2 VALUES(1, 2, 3);
|
sl@0
|
148 |
SELECT * from stuff;
|
sl@0
|
149 |
}
|
sl@0
|
150 |
} {abc square green}
|
sl@0
|
151 |
}
|
sl@0
|
152 |
|
sl@0
|
153 |
# UPDATE: The next test passes, but leaks memory. So leave it out.
|
sl@0
|
154 |
#
|
sl@0
|
155 |
# The following tests test that writing to the database from within
|
sl@0
|
156 |
# the xCommit callback causes a misuse error.
|
sl@0
|
157 |
# do_test vtab7-4.1 {
|
sl@0
|
158 |
# unset -nocomplain ::callbacks(xSync,abc)
|
sl@0
|
159 |
# set ::callbacks(xCommit,abc) {
|
sl@0
|
160 |
# execsql { INSERT INTO log VALUES('hello') }
|
sl@0
|
161 |
# }
|
sl@0
|
162 |
# catchsql {
|
sl@0
|
163 |
# INSERT INTO abc2 VALUES(1, 2, 3);
|
sl@0
|
164 |
# }
|
sl@0
|
165 |
# } {1 {library routine called out of sequence}}
|
sl@0
|
166 |
|
sl@0
|
167 |
# These tests, vtab7-4.*, test that an SQLITE_LOCKED error is returned
|
sl@0
|
168 |
# if an attempt to write to a virtual module table or create a new
|
sl@0
|
169 |
# virtual table from within an xSync() callback.
|
sl@0
|
170 |
do_test vtab7-4.1 {
|
sl@0
|
171 |
execsql {
|
sl@0
|
172 |
CREATE TABLE def(d, e, f);
|
sl@0
|
173 |
CREATE VIRTUAL TABLE def2 USING echo(def);
|
sl@0
|
174 |
}
|
sl@0
|
175 |
set ::callbacks(xSync,abc) {
|
sl@0
|
176 |
set ::error [catchsql { INSERT INTO def2 VALUES(1, 2, 3) }]
|
sl@0
|
177 |
}
|
sl@0
|
178 |
execsql {
|
sl@0
|
179 |
INSERT INTO abc2 VALUES(1, 2, 3);
|
sl@0
|
180 |
}
|
sl@0
|
181 |
set ::error
|
sl@0
|
182 |
} {1 {database table is locked}}
|
sl@0
|
183 |
do_test vtab7-4.2 {
|
sl@0
|
184 |
set ::callbacks(xSync,abc) {
|
sl@0
|
185 |
set ::error [catchsql { CREATE VIRTUAL TABLE def3 USING echo(def) }]
|
sl@0
|
186 |
}
|
sl@0
|
187 |
execsql {
|
sl@0
|
188 |
INSERT INTO abc2 VALUES(1, 2, 3);
|
sl@0
|
189 |
}
|
sl@0
|
190 |
set ::error
|
sl@0
|
191 |
} {1 {database table is locked}}
|
sl@0
|
192 |
|
sl@0
|
193 |
do_test vtab7-4.3 {
|
sl@0
|
194 |
set ::callbacks(xSync,abc) {
|
sl@0
|
195 |
set ::error [catchsql { DROP TABLE def2 }]
|
sl@0
|
196 |
}
|
sl@0
|
197 |
execsql {
|
sl@0
|
198 |
INSERT INTO abc2 VALUES(1, 2, 3);
|
sl@0
|
199 |
SELECT name FROM sqlite_master ORDER BY name;
|
sl@0
|
200 |
}
|
sl@0
|
201 |
set ::error
|
sl@0
|
202 |
} {1 {database table is locked}}
|
sl@0
|
203 |
|
sl@0
|
204 |
trace remove variable ::echo_module write echo_module_trace
|
sl@0
|
205 |
unset -nocomplain ::callbacks
|
sl@0
|
206 |
|
sl@0
|
207 |
finish_test
|