# 2022 December 5 # # 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. # #*********************************************************************** # set testdir [file dirname $argv0] source $testdir/tester.tcl set testprefix scanstatus2 ifcapable !scanstatus { finish_test return } sqlite3_db_config db STMT_SCANSTATUS 1 do_execsql_test 1.0 { CREATE TABLE t1(a, b); CREATE TABLE t2(x, y); INSERT INTO t1 VALUES(1, 2); INSERT INTO t1 VALUES(3, 4); INSERT INTO t2 VALUES('a', 'b'); INSERT INTO t2 VALUES('c', 'd'); INSERT INTO t2 VALUES('e', 'f'); } proc do_zexplain_test {v2 tn sql res} { db eval $sql set stmt [db version -last-stmt-ptr] set idx 0 set ret [list] set cmd sqlite3_stmt_scanstatus set f [list] if {$v2} { lappend f complex } while {1} { set r [sqlite3_stmt_scanstatus -flags $f $stmt $idx] if {[llength $r]==0} break lappend ret [dict get $r zExplain] incr idx } uplevel [list do_test $tn [list set {} $ret] [list {*}$res]] } proc get_cycles {stmt} { set r [sqlite3_stmt_scanstatus $stmt -1] dict get $r nCycle } proc foreach_scan {varname stmt body} { upvar $varname var for {set ii 0} {1} {incr ii} { set r [sqlite3_stmt_scanstatus -flags complex $stmt $ii] if {[llength $r]==0} break array set var $r uplevel $body } } proc get_eqp_graph {stmt iPar nIndent} { set res "" foreach_scan A $stmt { if {$A(iParentId)==$iPar} { set txt $A(zExplain) if {$A(nCycle)>=0} { append txt " (nCycle=$A(nCycle))" } append res "[string repeat - $nIndent]$txt\n" append res [get_eqp_graph $stmt $A(iSelectId) [expr $nIndent+2]] } } set res } proc get_graph {stmt} { set nCycle [get_cycles $stmt] set res "QUERY (nCycle=$nCycle)\n" append res [get_eqp_graph $stmt 0 2] } proc do_graph_test {tn sql res} { db eval $sql set stmt [db version -last-stmt-ptr] set graph [string trim [get_graph $stmt]] set graph [regsub -all {nCycle=[0-9]+} $graph nCycle=nnn] uplevel [list do_test $tn [list set {} $graph] [string trim $res]] } proc puts_graph {sql} { db eval $sql set stmt [db version -last-stmt-ptr] puts [string trim [get_graph $stmt]] } do_zexplain_test 0 1.1 { SELECT (SELECT a FROM t1 WHERE b=x) FROM t2 WHERE y=2 } { {SCAN t2} {SCAN t1} } do_zexplain_test 1 1.2 { SELECT (SELECT a FROM t1 WHERE b=x) FROM t2 WHERE y=2 } { {SCAN t2} {CORRELATED SCALAR SUBQUERY 1} {SCAN t1} } do_graph_test 1.3 { SELECT (SELECT a FROM t1 WHERE b=x) FROM t2 WHERE y=2 } { QUERY (nCycle=nnn) --SCAN t2 (nCycle=nnn) --CORRELATED SCALAR SUBQUERY 1 (nCycle=nnn) ----SCAN t1 (nCycle=nnn) } do_graph_test 1.4 { WITH v2(x,y) AS MATERIALIZED ( SELECT x,y FROM t2 ) SELECT * FROM t1, v2 ORDER BY y; } { QUERY (nCycle=nnn) --MATERIALIZE v2 (nCycle=nnn) ----SCAN t2 (nCycle=nnn) --SCAN v2 (nCycle=nnn) --SCAN t1 (nCycle=nnn) --USE TEMP B-TREE FOR ORDER BY (nCycle=nnn) } #------------------------------------------------------------------------- ifcapable fts5 { reset_db sqlite3_db_config db STMT_SCANSTATUS 1 do_execsql_test 2.0 { CREATE VIRTUAL TABLE ft USING fts5(a); INSERT INTO ft VALUES('abc'); INSERT INTO ft VALUES('def'); INSERT INTO ft VALUES('ghi'); } do_graph_test 2.1 { SELECT * FROM ft('def') } { QUERY (nCycle=nnn) --SCAN ft VIRTUAL TABLE INDEX 0:M1 (nCycle=nnn) } } #------------------------------------------------------------------------- reset_db sqlite3_db_config db STMT_SCANSTATUS 1 do_execsql_test 3.0 { CREATE TABLE x1(a, b); CREATE TABLE x2(c, d); WITH s(i) AS (SELECT 1 UNION ALL SELECT i+1 FROM s WHERE i<1000) INSERT INTO x1 SELECT i, i FROM s; INSERT INTO x2 SELECT a, b FROM x1; } do_graph_test 2.1 { SELECT * FROM x1, x2 WHERE c=+a; } { QUERY (nCycle=nnn) --SCAN x1 (nCycle=nnn) --CREATE AUTOMATIC INDEX ON x2(c, d) (nCycle=nnn) --BLOOM FILTER ON x2 (c=?) --SEARCH x2 USING AUTOMATIC COVERING INDEX (c=?) (nCycle=nnn) } #------------------------------------------------------------------------- reset_db sqlite3_db_config db STMT_SCANSTATUS 1 do_execsql_test 4.0 { CREATE TABLE rt1 (id INTEGER PRIMARY KEY, x1, x2); CREATE TABLE rt2 (id, x1, x2); } do_graph_test 4.1 { SELECT * FROM rt1, rt2 WHERE rt1.id%2 AND rt2.x1=rt1.x1; } { QUERY (nCycle=nnn) --SCAN rt1 (nCycle=nnn) --CREATE AUTOMATIC INDEX ON rt2(x1, id, x2) (nCycle=nnn) --BLOOM FILTER ON rt2 (x1=?) --SEARCH rt2 USING AUTOMATIC COVERING INDEX (x1=?) (nCycle=nnn) } do_graph_test 4.2 { SELECT rt2.id FROM rt1, rt2 WHERE rt1.id%2 AND rt2.x1=rt1.x1; } { QUERY (nCycle=nnn) --SCAN rt1 (nCycle=nnn) --CREATE AUTOMATIC INDEX ON rt2(x1, id) (nCycle=nnn) --BLOOM FILTER ON rt2 (x1=?) --SEARCH rt2 USING AUTOMATIC COVERING INDEX (x1=?) (nCycle=nnn) } do_graph_test 4.3 { SELECT rt2.id FROM rt1, rt2 WHERE rt1.id%2 AND (rt2.x1+1)=(rt1.x1+1); } { QUERY (nCycle=nnn) --SCAN rt1 (nCycle=nnn) --SCAN rt2 (nCycle=nnn) } do_graph_test 4.4 { SELECT rt2.id FROM rt1, rt2 WHERE rt1.id%2 AND rt2.x1=(rt1.x1+1) AND rt2.id>5; } { QUERY (nCycle=nnn) --SCAN rt1 (nCycle=nnn) --CREATE AUTOMATIC INDEX ON rt2(x1, id) WHERE <expr> (nCycle=nnn) --BLOOM FILTER ON rt2 (x1=?) --SEARCH rt2 USING AUTOMATIC PARTIAL COVERING INDEX (x1=?) (nCycle=nnn) } do_graph_test 4.5 { SELECT v1.cnt FROM rt1, ( SELECT count(*) AS cnt, rt2.x1 AS x1 FROM rt2 GROUP BY x1 ) AS v1 WHERE rt1.x1=v1.x1 } { QUERY (nCycle=nnn) --CO-ROUTINE v1 ----SCAN rt2 (nCycle=nnn) ----USE TEMP B-TREE FOR GROUP BY --SCAN rt1 (nCycle=nnn) --CREATE AUTOMATIC INDEX ON v1(x1, cnt) (nCycle=nnn) --BLOOM FILTER ON v1 (x1=?) --SEARCH v1 USING AUTOMATIC COVERING INDEX (x1=?) (nCycle=nnn) } #------------------------------------------------------------------------- reset_db ifcapable trace { do_execsql_test 5.0 { CREATE TABLE t1(x, y); CREATE TRIGGER tr1 AFTER DELETE ON t1 BEGIN SELECT 1; END; INSERT INTO t1 VALUES(1, 2); } proc trace {stmt sql} { array set A [sqlite3_stmt_scanstatus -flags complex [format %x $stmt] 0] lappend ::trace_explain $A(zExplain) } db trace_v2 trace set ::trace_explain [list] do_execsql_test 5.1 { DELETE FROM t1 WHERE x=1; } do_test 5.2 { set ::trace_explain } {{SCAN t1} {SCAN t1} {SCAN t1}} } finish_test