2017-10-11 21:00:34 +03:00
|
|
|
# 2017-10-11
|
|
|
|
#
|
|
|
|
# The author disclaims copyright to this source code. In place of
|
|
|
|
# a legal notice, here is a blessing:
|
|
|
|
#
|
|
|
|
# May you do good and not evil.
|
|
|
|
# May you find forgiveness for yourself and forgive others.
|
|
|
|
# May you share freely, never taking more than you give.
|
|
|
|
#
|
|
|
|
#***********************************************************************
|
|
|
|
# This file implements regression tests for SQLite library. The
|
|
|
|
# focus of this file is testing the checkfreelist extension.
|
|
|
|
#
|
|
|
|
|
|
|
|
set testdir [file dirname $argv0]
|
|
|
|
source $testdir/tester.tcl
|
|
|
|
set testprefix checkfreelist
|
|
|
|
|
|
|
|
ifcapable !vtab||!compound {
|
|
|
|
finish_test
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
if {[file exists ../checkfreelist.so]==0} {
|
|
|
|
finish_test
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
do_execsql_test 1.0 {
|
|
|
|
CREATE TABLE t1(a, b);
|
|
|
|
}
|
|
|
|
|
|
|
|
db enable_load_extension 1
|
|
|
|
do_execsql_test 1.1 {
|
|
|
|
SELECT load_extension('../checkfreelist.so');
|
|
|
|
} {{}}
|
|
|
|
|
|
|
|
do_execsql_test 1.2 { SELECT checkfreelist('main') } {ok}
|
|
|
|
do_execsql_test 1.3 {
|
|
|
|
WITH s(i) AS (
|
|
|
|
SELECT 1 UNION ALL SELECT i+1 FROM s WHERE i<10000
|
|
|
|
)
|
|
|
|
INSERT INTO t1 SELECT randomblob(400), randomblob(400) FROM s;
|
|
|
|
DELETE FROM t1 WHERE rowid%3;
|
|
|
|
PRAGMA freelist_count;
|
|
|
|
} {6726}
|
|
|
|
|
|
|
|
do_execsql_test 1.4 { SELECT checkfreelist('main') } {ok}
|
|
|
|
do_execsql_test 1.5 {
|
|
|
|
WITH freelist_trunk(i, d, n) AS (
|
|
|
|
SELECT 1, NULL, sqlite_readint32(data, 32) FROM sqlite_dbpage WHERE pgno=1
|
|
|
|
UNION ALL
|
|
|
|
SELECT n, data, sqlite_readint32(data)
|
|
|
|
FROM freelist_trunk, sqlite_dbpage WHERE pgno=n
|
|
|
|
)
|
|
|
|
SELECT i FROM freelist_trunk WHERE i!=1;
|
|
|
|
} {
|
|
|
|
10010 9716 9344 8970 8596 8223 7848 7475 7103 6728 6355 5983 5609 5235
|
|
|
|
4861 4488 4113 3741 3368 2993 2620 2248 1873 1500 1126 753 378 5
|
|
|
|
}
|
|
|
|
|
|
|
|
do_execsql_test 1.6 { SELECT checkfreelist('main') } {ok}
|
|
|
|
|
|
|
|
proc set_int {blob idx newval} {
|
|
|
|
binary scan $blob I* ints
|
|
|
|
lset ints $idx $newval
|
|
|
|
binary format I* $ints
|
|
|
|
}
|
|
|
|
db func set_int set_int
|
|
|
|
|
|
|
|
proc get_int {blob idx} {
|
|
|
|
binary scan $blob I* ints
|
|
|
|
lindex $ints $idx
|
|
|
|
}
|
|
|
|
db func get_int get_int
|
|
|
|
|
|
|
|
do_execsql_test 1.7 {
|
|
|
|
BEGIN;
|
|
|
|
UPDATE sqlite_dbpage
|
|
|
|
SET data = set_int(data, 1, get_int(data, 1)-1)
|
|
|
|
WHERE pgno=4861;
|
|
|
|
SELECT checkfreelist('main');
|
|
|
|
ROLLBACK;
|
|
|
|
} {{free-list count mismatch: actual=6725 header=6726}}
|
|
|
|
|
|
|
|
do_execsql_test 1.8 {
|
|
|
|
BEGIN;
|
|
|
|
UPDATE sqlite_dbpage
|
|
|
|
SET data = set_int(data, 5, (SELECT * FROM pragma_page_count)+1)
|
|
|
|
WHERE pgno=4861;
|
|
|
|
SELECT checkfreelist('main');
|
|
|
|
ROLLBACK;
|
|
|
|
} {{leaf page 10093 is out of range (child 3 of trunk page 4861)}}
|
|
|
|
|
|
|
|
do_execsql_test 1.9 {
|
|
|
|
BEGIN;
|
|
|
|
UPDATE sqlite_dbpage
|
|
|
|
SET data = set_int(data, 5, 0)
|
|
|
|
WHERE pgno=4861;
|
|
|
|
SELECT checkfreelist('main');
|
|
|
|
ROLLBACK;
|
|
|
|
} {{leaf page 0 is out of range (child 3 of trunk page 4861)}}
|
|
|
|
|
|
|
|
do_execsql_test 1.10 {
|
|
|
|
BEGIN;
|
|
|
|
UPDATE sqlite_dbpage
|
|
|
|
SET data = set_int(data, get_int(data, 1)+1, 0)
|
|
|
|
WHERE pgno=5;
|
|
|
|
SELECT checkfreelist('main');
|
|
|
|
ROLLBACK;
|
|
|
|
} {{leaf page 0 is out of range (child 247 of trunk page 5)}}
|
|
|
|
|
2017-10-11 21:21:44 +03:00
|
|
|
do_execsql_test 1.11 {
|
|
|
|
BEGIN;
|
|
|
|
UPDATE sqlite_dbpage
|
|
|
|
SET data = set_int(data, 1, 249)
|
|
|
|
WHERE pgno=5;
|
|
|
|
SELECT checkfreelist('main');
|
|
|
|
ROLLBACK;
|
|
|
|
} {{leaf count out of range (249) on trunk page 5}}
|
|
|
|
|
2017-10-11 21:00:34 +03:00
|
|
|
finish_test
|
|
|
|
|