1*c5c4113dSnw141292 2*c5c4113dSnw141292#pragma ident "%Z%%M% %I% %E% SMI" 3*c5c4113dSnw141292 4*c5c4113dSnw141292# 2002 July 17 5*c5c4113dSnw141292# 6*c5c4113dSnw141292# The author disclaims copyright to this source code. In place of 7*c5c4113dSnw141292# a legal notice, here is a blessing: 8*c5c4113dSnw141292# 9*c5c4113dSnw141292# May you do good and not evil. 10*c5c4113dSnw141292# May you find forgiveness for yourself and forgive others. 11*c5c4113dSnw141292# May you share freely, never taking more than you give. 12*c5c4113dSnw141292# 13*c5c4113dSnw141292#*********************************************************************** 14*c5c4113dSnw141292# This file implements regression tests for SQLite library. The 15*c5c4113dSnw141292# focus of this file is testing the ability of the library to detect 16*c5c4113dSnw141292# past or future file format version numbers and respond appropriately. 17*c5c4113dSnw141292# 18*c5c4113dSnw141292# $Id: version.test,v 1.9 2004/02/12 19:01:05 drh Exp $ 19*c5c4113dSnw141292 20*c5c4113dSnw141292set testdir [file dirname $argv0] 21*c5c4113dSnw141292source $testdir/tester.tcl 22*c5c4113dSnw141292 23*c5c4113dSnw141292# Current file format version 24*c5c4113dSnw141292set VX 4 25*c5c4113dSnw141292 26*c5c4113dSnw141292# Create a new database 27*c5c4113dSnw141292# 28*c5c4113dSnw141292do_test version-1.1 { 29*c5c4113dSnw141292 execsql { 30*c5c4113dSnw141292 CREATE TABLE t1(x); 31*c5c4113dSnw141292 INSERT INTO t1 VALUES(1); 32*c5c4113dSnw141292 INSERT INTO t1 SELECT x+1 FROM t1; 33*c5c4113dSnw141292 INSERT INTO t1 SELECT x+2 FROM t1; 34*c5c4113dSnw141292 INSERT INTO t1 SELECT x+4 FROM t1; 35*c5c4113dSnw141292 SELECT * FROM t1; 36*c5c4113dSnw141292 } 37*c5c4113dSnw141292} {1 2 3 4 5 6 7 8} 38*c5c4113dSnw141292 39*c5c4113dSnw141292# Make sure the version number is set correctly 40*c5c4113dSnw141292# 41*c5c4113dSnw141292do_test version-1.2 { 42*c5c4113dSnw141292 db close 43*c5c4113dSnw141292 set ::bt [btree_open test.db] 44*c5c4113dSnw141292 btree_begin_transaction $::bt 45*c5c4113dSnw141292 set ::meta [btree_get_meta $::bt] 46*c5c4113dSnw141292 btree_rollback $::bt 47*c5c4113dSnw141292 lindex $::meta 2 48*c5c4113dSnw141292} $VX 49*c5c4113dSnw141292 50*c5c4113dSnw141292# Increase the file_format number by one. Verify that the 51*c5c4113dSnw141292# file will refuse to open. 52*c5c4113dSnw141292# 53*c5c4113dSnw141292do_test version-1.3 { 54*c5c4113dSnw141292 set m2 [lreplace $::meta 2 2 [expr {$::VX+1}]] 55*c5c4113dSnw141292 btree_begin_transaction $::bt 56*c5c4113dSnw141292 eval btree_update_meta $::bt $m2 57*c5c4113dSnw141292 btree_commit $::bt 58*c5c4113dSnw141292 set rc [catch {sqlite db test.db} msg] 59*c5c4113dSnw141292 lappend rc $msg 60*c5c4113dSnw141292} {1 {unsupported file format}} 61*c5c4113dSnw141292 62*c5c4113dSnw141292# Decrease the file_format number by one. Verify that the 63*c5c4113dSnw141292# file will open correctly. 64*c5c4113dSnw141292# 65*c5c4113dSnw141292do_test version-1.4 { 66*c5c4113dSnw141292 set m2 [lreplace $::meta 2 2 [expr {$::VX-1}]] 67*c5c4113dSnw141292 btree_begin_transaction $::bt 68*c5c4113dSnw141292 eval btree_update_meta $::bt $m2 69*c5c4113dSnw141292 btree_commit $::bt 70*c5c4113dSnw141292 sqlite db test.db 71*c5c4113dSnw141292 execsql { 72*c5c4113dSnw141292 SELECT * FROM t1; 73*c5c4113dSnw141292 } 74*c5c4113dSnw141292} {1 2 3 4 5 6 7 8} 75*c5c4113dSnw141292 76*c5c4113dSnw141292# Set the file_format number to 2. This should cause the automatic 77*c5c4113dSnw141292# upgrade processing to run. 78*c5c4113dSnw141292# 79*c5c4113dSnw141292do_test version-1.5 { 80*c5c4113dSnw141292 set m2 [lreplace $::meta 2 2 2] 81*c5c4113dSnw141292 btree_begin_transaction $::bt 82*c5c4113dSnw141292 eval btree_update_meta $::bt $m2 83*c5c4113dSnw141292 btree_commit $::bt 84*c5c4113dSnw141292 sqlite db test.db 85*c5c4113dSnw141292 execsql { 86*c5c4113dSnw141292 SELECT * FROM t1; 87*c5c4113dSnw141292 } 88*c5c4113dSnw141292} {1 2 3 4 5 6 7 8} 89*c5c4113dSnw141292do_test version-1.6 { 90*c5c4113dSnw141292 set ::meta [btree_get_meta $::bt] 91*c5c4113dSnw141292 lindex $::meta 2 92*c5c4113dSnw141292} $VX 93*c5c4113dSnw141292 94*c5c4113dSnw141292# Add some triggers, views, and indices to the schema and make sure the 95*c5c4113dSnw141292# automatic upgrade still works. 96*c5c4113dSnw141292# 97*c5c4113dSnw141292do_test version-1.7 { 98*c5c4113dSnw141292 execsql { 99*c5c4113dSnw141292 CREATE INDEX i1 ON t1(x); 100*c5c4113dSnw141292 DELETE FROM t1; 101*c5c4113dSnw141292 CREATE TABLE t2(a INTEGER PRIMARY KEY, b UNIQUE, c); 102*c5c4113dSnw141292 CREATE TABLE cnt(name,ins, del); 103*c5c4113dSnw141292 INSERT INTO cnt VALUES('t1',0,0); 104*c5c4113dSnw141292 INSERT INTO cnt VALUES('t2',0,0); 105*c5c4113dSnw141292 CREATE TRIGGER r1 AFTER INSERT ON t1 FOR EACH ROW BEGIN 106*c5c4113dSnw141292 UPDATE cnt SET ins=ins+1 WHERE name='t1'; 107*c5c4113dSnw141292 END; 108*c5c4113dSnw141292 CREATE TRIGGER r2 AFTER DELETE ON t1 FOR EACH ROW BEGIN 109*c5c4113dSnw141292 UPDATE cnt SET del=del+1 WHERE name='t1'; 110*c5c4113dSnw141292 END; 111*c5c4113dSnw141292 CREATE TRIGGER r3 AFTER INSERT ON t2 FOR EACH ROW BEGIN 112*c5c4113dSnw141292 UPDATE cnt SET ins=ins+1 WHERE name='t2'; 113*c5c4113dSnw141292 END; 114*c5c4113dSnw141292 CREATE TRIGGER r4 AFTER DELETE ON t2 FOR EACH ROW BEGIN 115*c5c4113dSnw141292 UPDATE cnt SET del=del+1 WHERE name='t2'; 116*c5c4113dSnw141292 END; 117*c5c4113dSnw141292 CREATE VIEW v1 AS SELECT x+100 FROM t1; 118*c5c4113dSnw141292 CREATE VIEW v2 AS SELECT sum(ins), sum(del) FROM cnt; 119*c5c4113dSnw141292 INSERT INTO t1 VALUES(1); 120*c5c4113dSnw141292 INSERT INTO t1 SELECT x+1 FROM t1; 121*c5c4113dSnw141292 INSERT INTO t1 SELECT x+2 FROM t1; 122*c5c4113dSnw141292 INSERT INTO t1 SELECT x+4 FROM t1; 123*c5c4113dSnw141292 SELECT * FROM t1; 124*c5c4113dSnw141292 } 125*c5c4113dSnw141292} {1 2 3 4 5 6 7 8} 126*c5c4113dSnw141292do_test version-1.8 { 127*c5c4113dSnw141292 execsql { 128*c5c4113dSnw141292 SELECT * FROM v2; 129*c5c4113dSnw141292 } 130*c5c4113dSnw141292} {8 0} 131*c5c4113dSnw141292do_test version-1.9 { 132*c5c4113dSnw141292 execsql { 133*c5c4113dSnw141292 SELECT * FROM cnt; 134*c5c4113dSnw141292 } 135*c5c4113dSnw141292} {t1 8 0 t2 0 0} 136*c5c4113dSnw141292do_test version-1.10 { 137*c5c4113dSnw141292 execsql { 138*c5c4113dSnw141292 INSERT INTO t2 SELECT x*3, x*2, x FROM t1; 139*c5c4113dSnw141292 SELECT * FROM t2; 140*c5c4113dSnw141292 } 141*c5c4113dSnw141292} {3 2 1 6 4 2 9 6 3 12 8 4 15 10 5 18 12 6 21 14 7 24 16 8} 142*c5c4113dSnw141292do_test version-1.11 { 143*c5c4113dSnw141292 execsql { 144*c5c4113dSnw141292 SELECT * FROM cnt; 145*c5c4113dSnw141292 } 146*c5c4113dSnw141292} {t1 8 0 t2 8 0} 147*c5c4113dSnw141292 148*c5c4113dSnw141292# Here we do the upgrade test. 149*c5c4113dSnw141292# 150*c5c4113dSnw141292do_test version-1.12 { 151*c5c4113dSnw141292 db close 152*c5c4113dSnw141292 set m2 [lreplace $::meta 2 2 2] 153*c5c4113dSnw141292 btree_begin_transaction $::bt 154*c5c4113dSnw141292 eval btree_update_meta $::bt $m2 155*c5c4113dSnw141292 btree_commit $::bt 156*c5c4113dSnw141292 sqlite db test.db 157*c5c4113dSnw141292 execsql { 158*c5c4113dSnw141292 SELECT * FROM cnt; 159*c5c4113dSnw141292 } 160*c5c4113dSnw141292} {t1 8 0 t2 8 0} 161*c5c4113dSnw141292do_test version-1.13 { 162*c5c4113dSnw141292 execsql { 163*c5c4113dSnw141292 SELECT * FROM v1; 164*c5c4113dSnw141292 } 165*c5c4113dSnw141292} {101 102 103 104 105 106 107 108} 166*c5c4113dSnw141292do_test version-1.14 { 167*c5c4113dSnw141292 execsql { 168*c5c4113dSnw141292 SELECT * FROM v2; 169*c5c4113dSnw141292 } 170*c5c4113dSnw141292} {16 0} 171*c5c4113dSnw141292 172*c5c4113dSnw141292# Try to do an upgrade where the database file is read-only 173*c5c4113dSnw141292# 174*c5c4113dSnw141292do_test version-2.1 { 175*c5c4113dSnw141292 db close 176*c5c4113dSnw141292 set m2 [lreplace $::meta 2 2 2] 177*c5c4113dSnw141292 btree_begin_transaction $::bt 178*c5c4113dSnw141292 eval btree_update_meta $::bt $m2 179*c5c4113dSnw141292 btree_commit $::bt 180*c5c4113dSnw141292 btree_close $::bt 181*c5c4113dSnw141292 catch {file attributes test.db -permissions 0444} 182*c5c4113dSnw141292 catch {file attributes test.db -readonly 1} 183*c5c4113dSnw141292 if {[file writable test.db]} { 184*c5c4113dSnw141292 error "Unable to make the database file test.db readonly - rerun this test as an unprivileged user" 185*c5c4113dSnw141292 } 186*c5c4113dSnw141292 set rc [catch {sqlite db test.db} msg] 187*c5c4113dSnw141292 lappend rc $msg 188*c5c4113dSnw141292} {1 {unable to upgrade database to the version 2.6 format: attempt to write a readonly database}} 189*c5c4113dSnw141292do_test version-2.2 { 190*c5c4113dSnw141292 file delete -force test.db 191*c5c4113dSnw141292 set fd [open test.db w] 192*c5c4113dSnw141292 set txt "This is not a valid database file\n" 193*c5c4113dSnw141292 while {[string length $txt]<4092} {append txt $txt} 194*c5c4113dSnw141292 puts $fd $txt 195*c5c4113dSnw141292 close $fd 196*c5c4113dSnw141292 set rc [catch {sqlite db test.db} msg] 197*c5c4113dSnw141292 lappend rc $msg 198*c5c4113dSnw141292} {1 {file is encrypted or is not a database}} 199*c5c4113dSnw141292 200*c5c4113dSnw141292 201*c5c4113dSnw141292finish_test 202