| # 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)}} |
| |
| 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}} |
| |
| finish_test |
| |