| Index: third_party/sqlite/src/test/fts3aux1.test
|
| diff --git a/third_party/sqlite/src/test/fts3aux1.test b/third_party/sqlite/src/test/fts3aux1.test
|
| new file mode 100644
|
| index 0000000000000000000000000000000000000000..5359521ab7aa56d4e7fa63161c428401ebcea418
|
| --- /dev/null
|
| +++ b/third_party/sqlite/src/test/fts3aux1.test
|
| @@ -0,0 +1,462 @@
|
| +# 2011 January 27
|
| +#
|
| +# 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 script is testing the FTS3 module.
|
| +#
|
| +
|
| +set testdir [file dirname $argv0]
|
| +source $testdir/tester.tcl
|
| +ifcapable !fts3 { finish_test ; return }
|
| +set ::testprefix fts3aux1
|
| +
|
| +do_execsql_test 1.1 {
|
| + CREATE VIRTUAL TABLE t1 USING fts4;
|
| + INSERT INTO t1 VALUES('one two three four');
|
| + INSERT INTO t1 VALUES('three four five six');
|
| + INSERT INTO t1 VALUES('one three five seven');
|
| +
|
| + CREATE VIRTUAL TABLE terms USING fts4aux(t1);
|
| + SELECT term, documents, occurrences FROM terms WHERE col = '*';
|
| +} {
|
| + five 2 2 four 2 2 one 2 2 seven 1 1
|
| + six 1 1 three 3 3 two 1 1
|
| +}
|
| +
|
| +do_execsql_test 1.2 {
|
| + INSERT INTO t1 VALUES('one one one three three three');
|
| + SELECT term, documents, occurrences FROM terms WHERE col = '*';
|
| +} {
|
| + five 2 2 four 2 2 one 3 5 seven 1 1
|
| + six 1 1 three 4 6 two 1 1
|
| +}
|
| +
|
| +do_execsql_test 1.3 {
|
| + DELETE FROM t1;
|
| + SELECT term, documents, occurrences FROM terms WHERE col = '*';
|
| +} {}
|
| +
|
| +do_execsql_test 1.4 {
|
| + INSERT INTO t1 VALUES('a b a b a b a');
|
| + INSERT INTO t1 SELECT * FROM t1;
|
| + INSERT INTO t1 SELECT * FROM t1;
|
| + INSERT INTO t1 SELECT * FROM t1;
|
| + INSERT INTO t1 SELECT * FROM t1;
|
| + INSERT INTO t1 SELECT * FROM t1;
|
| + INSERT INTO t1 SELECT * FROM t1;
|
| + INSERT INTO t1 SELECT * FROM t1;
|
| + INSERT INTO t1 SELECT * FROM t1;
|
| + SELECT term, documents, occurrences FROM terms WHERE col = '*';
|
| +} {a 256 1024 b 256 768}
|
| +
|
| +#-------------------------------------------------------------------------
|
| +# The following tests verify that the fts4aux module uses the full-text
|
| +# index to reduce the number of rows scanned in the following circumstances:
|
| +#
|
| +# * when there is equality comparison against the term column using the
|
| +# BINARY collating sequence.
|
| +#
|
| +# * when there is a range constraint on the term column using the BINARY
|
| +# collating sequence.
|
| +#
|
| +# And also uses the full-text index to optimize ORDER BY clauses of the
|
| +# form "ORDER BY term ASC" or equivalent.
|
| +#
|
| +# Test organization is:
|
| +#
|
| +# fts3aux1-2.1.*: equality constraints.
|
| +# fts3aux1-2.2.*: range constraints.
|
| +# fts3aux1-2.3.*: ORDER BY optimization.
|
| +#
|
| +
|
| +do_execsql_test 2.0 {
|
| + DROP TABLE t1;
|
| + DROP TABLE terms;
|
| +
|
| + CREATE VIRTUAL TABLE x1 USING fts4(x);
|
| + INSERT INTO x1(x1) VALUES('nodesize=24');
|
| + CREATE VIRTUAL TABLE terms USING fts4aux(x1);
|
| +
|
| + CREATE VIEW terms_v AS
|
| + SELECT term, documents, occurrences FROM terms WHERE col = '*';
|
| +
|
| + INSERT INTO x1 VALUES('braes brag bragged bragger bragging');
|
| + INSERT INTO x1 VALUES('brags braid braided braiding braids');
|
| + INSERT INTO x1 VALUES('brain brainchild brained braining brains');
|
| + INSERT INTO x1 VALUES('brainstem brainstems brainstorm brainstorms');
|
| +}
|
| +
|
| +proc rec {varname x} {
|
| + global $varname
|
| + incr $varname
|
| + return 1
|
| +}
|
| +db func rec rec
|
| +
|
| +# Use EQP to show that the WHERE expression "term='braid'" uses a different
|
| +# index number (1) than "+term='braid'" (0).
|
| +#
|
| +do_execsql_test 2.1.1.1 {
|
| + EXPLAIN QUERY PLAN SELECT * FROM terms WHERE term='braid'
|
| +} { 0 0 0 {SCAN TABLE terms VIRTUAL TABLE INDEX 1: (~0 rows)} }
|
| +do_execsql_test 2.1.1.2 {
|
| + EXPLAIN QUERY PLAN SELECT * FROM terms WHERE +term='braid'
|
| +} {0 0 0 {SCAN TABLE terms VIRTUAL TABLE INDEX 0: (~0 rows)}}
|
| +
|
| +# Now show that using "term='braid'" means the virtual table returns
|
| +# only 1 row to SQLite, but "+term='braid'" means all 19 are returned.
|
| +#
|
| +do_test 2.1.2.1 {
|
| + set cnt 0
|
| + execsql { SELECT * FROM terms_v WHERE rec('cnt', term) AND term='braid' }
|
| + set cnt
|
| +} {2}
|
| +do_test 2.1.2.2 {
|
| + set cnt 0
|
| + execsql { SELECT * FROM terms_v WHERE rec('cnt', term) AND +term='braid' }
|
| + set cnt
|
| +} {38}
|
| +
|
| +# Similar to the test immediately above, but using a term ("breakfast") that
|
| +# is not featured in the dataset.
|
| +#
|
| +do_test 2.1.3.1 {
|
| + set cnt 0
|
| + execsql { SELECT * FROM terms_v WHERE rec('cnt', term) AND term='breakfast' }
|
| + set cnt
|
| +} {0}
|
| +do_test 2.1.3.2 {
|
| + set cnt 0
|
| + execsql { SELECT * FROM terms_v WHERE rec('cnt', term) AND +term='breakfast' }
|
| + set cnt
|
| +} {38}
|
| +
|
| +do_execsql_test 2.1.4.1 { SELECT * FROM terms_v WHERE term='braid' } {braid 1 1}
|
| +do_execsql_test 2.1.4.2 { SELECT * FROM terms_v WHERE +term='braid'} {braid 1 1}
|
| +do_execsql_test 2.1.4.3 { SELECT * FROM terms_v WHERE term='breakfast' } {}
|
| +do_execsql_test 2.1.4.4 { SELECT * FROM terms_v WHERE +term='breakfast' } {}
|
| +
|
| +do_execsql_test 2.1.4.5 { SELECT * FROM terms_v WHERE term='cba' } {}
|
| +do_execsql_test 2.1.4.6 { SELECT * FROM terms_v WHERE +term='cba' } {}
|
| +do_execsql_test 2.1.4.7 { SELECT * FROM terms_v WHERE term='abc' } {}
|
| +do_execsql_test 2.1.4.8 { SELECT * FROM terms_v WHERE +term='abc' } {}
|
| +
|
| +# Special case: term=NULL
|
| +#
|
| +do_execsql_test 2.1.5 { SELECT * FROM terms WHERE term=NULL } {}
|
| +
|
| +do_execsql_test 2.2.1.1 {
|
| + EXPLAIN QUERY PLAN SELECT * FROM terms WHERE term>'brain'
|
| +} { 0 0 0 {SCAN TABLE terms VIRTUAL TABLE INDEX 2: (~0 rows)} }
|
| +do_execsql_test 2.2.1.2 {
|
| + EXPLAIN QUERY PLAN SELECT * FROM terms WHERE +term>'brain'
|
| +} { 0 0 0 {SCAN TABLE terms VIRTUAL TABLE INDEX 0: (~0 rows)} }
|
| +
|
| +do_execsql_test 2.2.1.3 {
|
| + EXPLAIN QUERY PLAN SELECT * FROM terms WHERE term<'brain'
|
| +} { 0 0 0 {SCAN TABLE terms VIRTUAL TABLE INDEX 4: (~0 rows)} }
|
| +do_execsql_test 2.2.1.4 {
|
| + EXPLAIN QUERY PLAN SELECT * FROM terms WHERE +term<'brain'
|
| +} { 0 0 0 {SCAN TABLE terms VIRTUAL TABLE INDEX 0: (~0 rows)} }
|
| +
|
| +do_execsql_test 2.2.1.5 {
|
| + EXPLAIN QUERY PLAN SELECT * FROM terms WHERE term BETWEEN 'brags' AND 'brain'
|
| +} { 0 0 0 {SCAN TABLE terms VIRTUAL TABLE INDEX 6: (~0 rows)} }
|
| +do_execsql_test 2.2.1.6 {
|
| + EXPLAIN QUERY PLAN SELECT * FROM terms WHERE +term BETWEEN 'brags' AND 'brain'
|
| +} { 0 0 0 {SCAN TABLE terms VIRTUAL TABLE INDEX 0: (~0 rows)} }
|
| +
|
| +do_test 2.2.2.1 {
|
| + set cnt 0
|
| + execsql { SELECT * FROM terms WHERE rec('cnt', term) AND term>'brain' }
|
| + set cnt
|
| +} {18}
|
| +do_test 2.2.2.2 {
|
| + set cnt 0
|
| + execsql { SELECT * FROM terms WHERE rec('cnt', term) AND +term>'brain' }
|
| + set cnt
|
| +} {38}
|
| +do_execsql_test 2.2.2.3 {
|
| + SELECT term, documents, occurrences FROM terms_v WHERE term>'brain'
|
| +} {
|
| + brainchild 1 1 brained 1 1 braining 1 1 brains 1 1
|
| + brainstem 1 1 brainstems 1 1 brainstorm 1 1 brainstorms 1 1
|
| +}
|
| +do_execsql_test 2.2.2.4 {
|
| + SELECT term, documents, occurrences FROM terms_v WHERE +term>'brain'
|
| +} {
|
| + brainchild 1 1 brained 1 1 braining 1 1 brains 1 1
|
| + brainstem 1 1 brainstems 1 1 brainstorm 1 1 brainstorms 1 1
|
| +}
|
| +do_execsql_test 2.2.2.5 {
|
| + SELECT term, documents, occurrences FROM terms_v WHERE term>='brain'
|
| +} {
|
| + brain 1 1
|
| + brainchild 1 1 brained 1 1 braining 1 1 brains 1 1
|
| + brainstem 1 1 brainstems 1 1 brainstorm 1 1 brainstorms 1 1
|
| +}
|
| +do_execsql_test 2.2.2.6 {
|
| + SELECT term, documents, occurrences FROM terms_v WHERE +term>='brain'
|
| +} {
|
| + brain 1 1
|
| + brainchild 1 1 brained 1 1 braining 1 1 brains 1 1
|
| + brainstem 1 1 brainstems 1 1 brainstorm 1 1 brainstorms 1 1
|
| +}
|
| +
|
| +do_execsql_test 2.2.2.7 {
|
| + SELECT term, documents, occurrences FROM terms_v WHERE term>='abc'
|
| +} {
|
| + braes 1 1 brag 1 1 bragged 1 1 bragger 1 1
|
| + bragging 1 1 brags 1 1 braid 1 1 braided 1 1
|
| + braiding 1 1 braids 1 1 brain 1 1 brainchild 1 1
|
| + brained 1 1 braining 1 1 brains 1 1 brainstem 1 1
|
| + brainstems 1 1 brainstorm 1 1 brainstorms 1 1
|
| +}
|
| +do_execsql_test 2.2.2.8 {
|
| + SELECT term, documents, occurrences FROM terms_v WHERE +term>='abc'
|
| +} {
|
| + braes 1 1 brag 1 1 bragged 1 1 bragger 1 1
|
| + bragging 1 1 brags 1 1 braid 1 1 braided 1 1
|
| + braiding 1 1 braids 1 1 brain 1 1 brainchild 1 1
|
| + brained 1 1 braining 1 1 brains 1 1 brainstem 1 1
|
| + brainstems 1 1 brainstorm 1 1 brainstorms 1 1
|
| +}
|
| +
|
| +do_execsql_test 2.2.2.9 {
|
| + SELECT term, documents, occurrences FROM terms_v WHERE term>='brainstorms'
|
| +} {brainstorms 1 1}
|
| +do_execsql_test 2.2.2.10 {
|
| + SELECT term, documents, occurrences FROM terms_v WHERE term>='brainstorms'
|
| +} {brainstorms 1 1}
|
| +do_execsql_test 2.2.2.11 { SELECT * FROM terms_v WHERE term>'brainstorms' } {}
|
| +do_execsql_test 2.2.2.12 { SELECT * FROM terms_v WHERE term>'brainstorms' } {}
|
| +
|
| +do_execsql_test 2.2.2.13 { SELECT * FROM terms_v WHERE term>'cba' } {}
|
| +do_execsql_test 2.2.2.14 { SELECT * FROM terms_v WHERE term>'cba' } {}
|
| +
|
| +do_test 2.2.3.1 {
|
| + set cnt 0
|
| + execsql { SELECT * FROM terms WHERE rec('cnt', term) AND term<'brain' }
|
| + set cnt
|
| +} {22}
|
| +do_test 2.2.3.2 {
|
| + set cnt 0
|
| + execsql { SELECT * FROM terms WHERE rec('cnt', term) AND +term<'brain' }
|
| + set cnt
|
| +} {38}
|
| +do_execsql_test 2.2.3.3 {
|
| + SELECT term, documents, occurrences FROM terms_v WHERE term<'brain'
|
| +} {
|
| + braes 1 1 brag 1 1 bragged 1 1 bragger 1 1 bragging 1 1
|
| + brags 1 1 braid 1 1 braided 1 1 braiding 1 1 braids 1 1
|
| +}
|
| +do_execsql_test 2.2.3.4 {
|
| + SELECT term, documents, occurrences FROM terms_v WHERE +term<'brain'
|
| +} {
|
| + braes 1 1 brag 1 1 bragged 1 1 bragger 1 1 bragging 1 1
|
| + brags 1 1 braid 1 1 braided 1 1 braiding 1 1 braids 1 1
|
| +}
|
| +do_execsql_test 2.2.3.5 {
|
| + SELECT term, documents, occurrences FROM terms_v WHERE term<='brain'
|
| +} {
|
| + braes 1 1 brag 1 1 bragged 1 1 bragger 1 1 bragging 1 1
|
| + brags 1 1 braid 1 1 braided 1 1 braiding 1 1 braids 1 1
|
| + brain 1 1
|
| +}
|
| +do_execsql_test 2.2.3.6 {
|
| + SELECT term, documents, occurrences FROM terms_v WHERE +term<='brain'
|
| +} {
|
| + braes 1 1 brag 1 1 bragged 1 1 bragger 1 1 bragging 1 1
|
| + brags 1 1 braid 1 1 braided 1 1 braiding 1 1 braids 1 1
|
| + brain 1 1
|
| +}
|
| +
|
| +do_test 2.2.4.1 {
|
| + set cnt 0
|
| + execsql {
|
| + SELECT term, documents, occurrences FROM terms
|
| + WHERE rec('cnt', term) AND term BETWEEN 'brags' AND 'brain'
|
| + }
|
| + set cnt
|
| +} {12}
|
| +do_test 2.2.4.2 {
|
| + set cnt 0
|
| + execsql {
|
| + SELECT term, documents, occurrences FROM terms
|
| + WHERE rec('cnt', term) AND +term BETWEEN 'brags' AND 'brain'
|
| + }
|
| + set cnt
|
| +} {38}
|
| +do_execsql_test 2.2.4.3 {
|
| + SELECT term, documents, occurrences FROM terms_v
|
| + WHERE rec('cnt', term) AND term BETWEEN 'brags' AND 'brain'
|
| +} {
|
| + brags 1 1 braid 1 1 braided 1 1 braiding 1 1 braids 1 1 brain 1 1
|
| +}
|
| +do_execsql_test 2.2.4.4 {
|
| + SELECT term, documents, occurrences FROM terms_v
|
| + WHERE rec('cnt', term) AND +term BETWEEN 'brags' AND 'brain'
|
| +} {
|
| + brags 1 1 braid 1 1 braided 1 1 braiding 1 1 braids 1 1 brain 1 1
|
| +}
|
| +do_execsql_test 2.2.4.5 {
|
| + SELECT term, documents, occurrences FROM terms_v
|
| + WHERE rec('cnt', term) AND term > 'brags' AND term < 'brain'
|
| +} {
|
| + braid 1 1 braided 1 1 braiding 1 1 braids 1 1
|
| +}
|
| +do_execsql_test 2.2.4.6 {
|
| + SELECT term, documents, occurrences FROM terms_v
|
| + WHERE rec('cnt', term) AND +term > 'brags' AND +term < 'brain'
|
| +} {
|
| + braid 1 1 braided 1 1 braiding 1 1 braids 1 1
|
| +}
|
| +
|
| +# Check that "ORDER BY term ASC" and equivalents are sorted by the
|
| +# virtual table implementation. Any other ORDER BY clause requires
|
| +# SQLite to sort results using a temporary b-tree.
|
| +#
|
| +foreach {tn sort orderby} {
|
| + 1 0 "ORDER BY term ASC"
|
| + 2 0 "ORDER BY term"
|
| + 3 1 "ORDER BY term DESC"
|
| + 4 1 "ORDER BY documents ASC"
|
| + 5 1 "ORDER BY documents"
|
| + 6 1 "ORDER BY documents DESC"
|
| + 7 1 "ORDER BY occurrences ASC"
|
| + 8 1 "ORDER BY occurrences"
|
| + 9 1 "ORDER BY occurrences DESC"
|
| +} {
|
| +
|
| + set res [list 0 0 0 {SCAN TABLE terms VIRTUAL TABLE INDEX 0: (~0 rows)}]
|
| + if {$sort} { lappend res 0 0 0 {USE TEMP B-TREE FOR ORDER BY} }
|
| +
|
| + set sql "SELECT * FROM terms $orderby"
|
| + do_execsql_test 2.3.1.$tn "EXPLAIN QUERY PLAN $sql" $res
|
| +}
|
| +
|
| +#-------------------------------------------------------------------------
|
| +# The next set of tests, fts3aux1-3.*, test error conditions in the
|
| +# fts4aux module. Except, fault injection testing (OOM, IO error etc.) is
|
| +# done in fts3fault2.test
|
| +#
|
| +
|
| +do_execsql_test 3.1.1 {
|
| + CREATE VIRTUAL TABLE t2 USING fts4;
|
| +}
|
| +
|
| +do_catchsql_test 3.1.2 {
|
| + CREATE VIRTUAL TABLE terms2 USING fts4aux;
|
| +} {1 {wrong number of arguments to fts4aux constructor}}
|
| +do_catchsql_test 3.1.3 {
|
| + CREATE VIRTUAL TABLE terms2 USING fts4aux(t2, t2);
|
| +} {1 {wrong number of arguments to fts4aux constructor}}
|
| +
|
| +do_execsql_test 3.2.1 {
|
| + CREATE VIRTUAL TABLE terms3 USING fts4aux(does_not_exist)
|
| +}
|
| +do_catchsql_test 3.2.2 {
|
| + SELECT * FROM terms3
|
| +} {1 {SQL logic error or missing database}}
|
| +do_catchsql_test 3.2.3 {
|
| + SELECT * FROM terms3 WHERE term = 'abc'
|
| +} {1 {SQL logic error or missing database}}
|
| +
|
| +do_catchsql_test 3.3.1 {
|
| + INSERT INTO terms VALUES(1,2,3);
|
| +} {1 {table terms may not be modified}}
|
| +do_catchsql_test 3.3.2 {
|
| + DELETE FROM terms
|
| +} {1 {table terms may not be modified}}
|
| +do_catchsql_test 3.3.3 {
|
| + UPDATE terms set documents = documents+1;
|
| +} {1 {table terms may not be modified}}
|
| +
|
| +
|
| +#-------------------------------------------------------------------------
|
| +# The following tests - fts4aux-4.* - test that joins work with fts4aux
|
| +# tables. And that fts4aux provides reasonably sane cost information via
|
| +# xBestIndex to the query planner.
|
| +#
|
| +db close
|
| +forcedelete test.db
|
| +sqlite3 db test.db
|
| +do_execsql_test 4.1 {
|
| + CREATE VIRTUAL TABLE x1 USING fts4(x);
|
| + CREATE VIRTUAL TABLE terms USING fts4aux(x1);
|
| + CREATE TABLE x2(y);
|
| + CREATE TABLE x3(y);
|
| + CREATE INDEX i1 ON x3(y);
|
| +
|
| + INSERT INTO x1 VALUES('a b c d e');
|
| + INSERT INTO x1 VALUES('f g h i j');
|
| + INSERT INTO x1 VALUES('k k l l a');
|
| +
|
| + INSERT INTO x2 SELECT term FROM terms WHERE col = '*';
|
| + INSERT INTO x3 SELECT term FROM terms WHERE col = '*';
|
| +}
|
| +
|
| +proc do_plansql_test {tn sql r} {
|
| + uplevel do_execsql_test $tn [list "EXPLAIN QUERY PLAN $sql ; $sql"] [list $r]
|
| +}
|
| +
|
| +do_plansql_test 4.2 {
|
| + SELECT y FROM x2, terms WHERE y = term AND col = '*'
|
| +} {
|
| + 0 0 0 {SCAN TABLE x2 (~1000000 rows)}
|
| + 0 1 1 {SCAN TABLE terms VIRTUAL TABLE INDEX 1: (~0 rows)}
|
| + a b c d e f g h i j k l
|
| +}
|
| +
|
| +do_plansql_test 4.3 {
|
| + SELECT y FROM terms, x2 WHERE y = term AND col = '*'
|
| +} {
|
| + 0 0 1 {SCAN TABLE x2 (~1000000 rows)}
|
| + 0 1 0 {SCAN TABLE terms VIRTUAL TABLE INDEX 1: (~0 rows)}
|
| + a b c d e f g h i j k l
|
| +}
|
| +
|
| +do_plansql_test 4.4 {
|
| + SELECT y FROM x3, terms WHERE y = term AND col = '*'
|
| +} {
|
| + 0 0 1 {SCAN TABLE terms VIRTUAL TABLE INDEX 0: (~0 rows)}
|
| + 0 1 0 {SEARCH TABLE x3 USING COVERING INDEX i1 (y=?) (~10 rows)}
|
| + a b c d e f g h i j k l
|
| +}
|
| +
|
| +do_plansql_test 4.5 {
|
| + SELECT y FROM terms, x3 WHERE y = term AND occurrences>1 AND col = '*'
|
| +} {
|
| + 0 0 0 {SCAN TABLE terms VIRTUAL TABLE INDEX 0: (~0 rows)}
|
| + 0 1 1 {SEARCH TABLE x3 USING COVERING INDEX i1 (y=?) (~10 rows)}
|
| + a k l
|
| +}
|
| +
|
| +#-------------------------------------------------------------------------
|
| +# The following tests check that fts4aux can handle an fts table with an
|
| +# odd name (one that requires quoting for use in SQL statements). And that
|
| +# the argument to the fts4aux constructor is properly dequoted before use.
|
| +#
|
| +#
|
| +do_execsql_test 5.1 {
|
| + CREATE VIRTUAL TABLE "abc '!' def" USING fts4(x, y);
|
| + INSERT INTO "abc '!' def" VALUES('XX', 'YY');
|
| +
|
| + CREATE VIRTUAL TABLE terms3 USING fts4aux("abc '!' def");
|
| + SELECT * FROM terms3;
|
| +} {xx * 1 1 xx 0 1 1 yy * 1 1 yy 1 1 1}
|
| +
|
| +do_execsql_test 5.2 {
|
| + CREATE VIRTUAL TABLE "%%^^%%" USING fts4aux('abc ''!'' def');
|
| + SELECT * FROM "%%^^%%";
|
| +} {xx * 1 1 xx 0 1 1 yy * 1 1 yy 1 1 1}
|
| +
|
| +
|
| +finish_test
|
|
|