2018-06-15 22:01:35 +03:00
|
|
|
# 2018 May 8
|
|
|
|
#
|
|
|
|
# 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.
|
|
|
|
#
|
|
|
|
|
|
|
|
set testdir [file dirname $argv0]
|
|
|
|
source $testdir/tester.tcl
|
|
|
|
set testprefix windowfault
|
|
|
|
|
2018-06-22 23:51:35 +03:00
|
|
|
ifcapable !windowfunc {
|
|
|
|
finish_test
|
|
|
|
return
|
|
|
|
}
|
2018-06-15 22:01:35 +03:00
|
|
|
|
|
|
|
do_execsql_test 1.0 {
|
|
|
|
CREATE TABLE t1(a, b, c, d);
|
|
|
|
INSERT INTO t1 VALUES(1, 2, 3, 4);
|
|
|
|
INSERT INTO t1 VALUES(5, 6, 7, 8);
|
|
|
|
INSERT INTO t1 VALUES(9, 10, 11, 12);
|
|
|
|
}
|
|
|
|
faultsim_save_and_close
|
|
|
|
|
2019-03-18 21:55:35 +03:00
|
|
|
do_faultsim_test 1 -start 1 -faults oom-t* -prep {
|
2018-06-15 22:01:35 +03:00
|
|
|
faultsim_restore_and_reopen
|
|
|
|
} -body {
|
|
|
|
execsql {
|
|
|
|
SELECT row_number() OVER win,
|
|
|
|
rank() OVER win,
|
|
|
|
dense_rank() OVER win,
|
|
|
|
ntile(2) OVER win,
|
|
|
|
first_value(d) OVER win,
|
|
|
|
last_value(d) OVER win,
|
|
|
|
nth_value(d,2) OVER win,
|
|
|
|
lead(d) OVER win,
|
|
|
|
lag(d) OVER win,
|
|
|
|
max(d) OVER win,
|
|
|
|
min(d) OVER win
|
|
|
|
FROM t1
|
|
|
|
WINDOW win AS (ORDER BY a)
|
|
|
|
}
|
|
|
|
} -test {
|
|
|
|
faultsim_test_result {0 {1 1 1 1 4 4 {} 8 {} 4 4 2 2 2 1 4 8 8 12 4 8 4 3 3 3 2 4 12 8 {} 8 12 4}}
|
|
|
|
}
|
|
|
|
|
2018-06-20 12:23:49 +03:00
|
|
|
do_faultsim_test 1.1 -faults oom-t* -prep {
|
|
|
|
faultsim_restore_and_reopen
|
|
|
|
} -body {
|
|
|
|
execsql {
|
|
|
|
SELECT row_number() OVER win,
|
|
|
|
rank() OVER win,
|
|
|
|
dense_rank() OVER win
|
|
|
|
FROM t1
|
|
|
|
WINDOW win AS (PARTITION BY c<7 ORDER BY a)
|
|
|
|
}
|
|
|
|
} -test {
|
|
|
|
faultsim_test_result {0 {1 1 1 2 2 2 1 1 1}}
|
|
|
|
}
|
|
|
|
|
2018-06-22 20:57:10 +03:00
|
|
|
do_faultsim_test 1.2 -faults oom-t* -prep {
|
|
|
|
faultsim_restore_and_reopen
|
|
|
|
} -body {
|
|
|
|
execsql {
|
|
|
|
SELECT ntile(105)
|
|
|
|
OVER ( RANGE BETWEEN UNBOUNDED PRECEDING AND CURRENT ROW )
|
|
|
|
FROM t1
|
|
|
|
}
|
|
|
|
} -test {
|
|
|
|
faultsim_test_result {0 {1 2 3}}
|
|
|
|
}
|
|
|
|
|
2018-06-20 12:23:49 +03:00
|
|
|
do_faultsim_test 2 -start 1 -faults oom-* -prep {
|
|
|
|
faultsim_restore_and_reopen
|
|
|
|
} -body {
|
|
|
|
execsql {
|
|
|
|
SELECT round(percent_rank() OVER win, 2),
|
|
|
|
round(cume_dist() OVER win, 2)
|
|
|
|
FROM t1
|
|
|
|
WINDOW win AS (ORDER BY a)
|
|
|
|
}
|
|
|
|
} -test {
|
|
|
|
faultsim_test_result {0 {0.0 0.33 0.5 0.67 1.0 1.0}}
|
|
|
|
}
|
|
|
|
|
|
|
|
do_faultsim_test 3 -faults oom-* -prep {
|
2018-06-15 23:46:12 +03:00
|
|
|
faultsim_restore_and_reopen
|
|
|
|
} -body {
|
|
|
|
execsql {
|
|
|
|
SELECT min(d) OVER win, max(d) OVER win
|
|
|
|
FROM t1
|
|
|
|
WINDOW win AS (ORDER BY a RANGE BETWEEN CURRENT ROW AND UNBOUNDED FOLLOWING)
|
|
|
|
}
|
|
|
|
} -test {
|
|
|
|
faultsim_test_result {0 {4 12 8 12 12 12}}
|
|
|
|
}
|
|
|
|
|
2018-06-20 12:23:49 +03:00
|
|
|
do_faultsim_test 4 -faults oom-* -prep {
|
|
|
|
faultsim_restore_and_reopen
|
|
|
|
} -body {
|
|
|
|
execsql {
|
|
|
|
CREATE VIEW aaa AS
|
|
|
|
SELECT min(d) OVER w, max(d) OVER w
|
|
|
|
FROM t1
|
|
|
|
WINDOW w AS (ORDER BY a RANGE BETWEEN CURRENT ROW AND UNBOUNDED FOLLOWING);
|
|
|
|
SELECT * FROM aaa;
|
|
|
|
}
|
|
|
|
} -test {
|
|
|
|
faultsim_test_result {0 {4 12 8 12 12 12}}
|
|
|
|
}
|
|
|
|
|
|
|
|
do_faultsim_test 5 -start 1 -faults oom-* -prep {
|
|
|
|
faultsim_restore_and_reopen
|
|
|
|
} -body {
|
|
|
|
execsql {
|
|
|
|
SELECT last_value(a) OVER win1,
|
|
|
|
last_value(a) OVER win2
|
|
|
|
FROM t1
|
|
|
|
WINDOW win1 AS (ORDER BY a ROWS BETWEEN CURRENT ROW AND 1 FOLLOWING),
|
|
|
|
win2 AS (ORDER BY a)
|
|
|
|
}
|
|
|
|
} -test {
|
|
|
|
faultsim_test_result {0 {5 1 9 5 9 9}}
|
|
|
|
}
|
2018-06-15 22:01:35 +03:00
|
|
|
|
2018-07-06 16:25:02 +03:00
|
|
|
do_faultsim_test 6 -faults oom-* -prep {
|
|
|
|
faultsim_restore_and_reopen
|
|
|
|
} -body {
|
|
|
|
execsql {
|
|
|
|
SELECT percent_rank() OVER (), cume_dist() OVER () FROM t1
|
|
|
|
}
|
|
|
|
} -test {
|
|
|
|
faultsim_test_result {0 {0.0 1.0 0.0 1.0 0.0 1.0}}
|
|
|
|
}
|
|
|
|
|
2018-07-09 21:11:52 +03:00
|
|
|
do_faultsim_test 7 -faults oom-* -prep {
|
|
|
|
faultsim_restore_and_reopen
|
|
|
|
} -body {
|
|
|
|
execsql {
|
|
|
|
SELECT percent_rank() OVER (), cume_dist() OVER () FROM t1
|
|
|
|
}
|
|
|
|
} -test {
|
|
|
|
faultsim_test_result {0 {0.0 1.0 0.0 1.0 0.0 1.0}}
|
|
|
|
}
|
|
|
|
|
|
|
|
do_faultsim_test 8 -faults oom-t* -prep {
|
|
|
|
faultsim_restore_and_reopen
|
|
|
|
} -body {
|
|
|
|
execsql {
|
|
|
|
SELECT a, sum(b) OVER win1 FROM t1
|
|
|
|
WINDOW win1 AS (PARTITION BY a ),
|
|
|
|
win2 AS (PARTITION BY b )
|
|
|
|
ORDER BY a;
|
|
|
|
}
|
|
|
|
} -test {
|
|
|
|
faultsim_test_result {0 {1 2 5 6 9 10}}
|
|
|
|
}
|
|
|
|
|
2019-03-30 20:07:23 +03:00
|
|
|
#-------------------------------------------------------------------------
|
|
|
|
# The following test causes a cursor in REQURESEEK state to be passed
|
|
|
|
# to sqlite3BtreeDelete(). An error is simulated within the seek operation
|
|
|
|
# to restore the cursors position.
|
|
|
|
#
|
|
|
|
reset_db
|
|
|
|
set big [string repeat x 900]
|
|
|
|
do_execsql_test 9.0 {
|
|
|
|
PRAGMA page_size = 512;
|
|
|
|
PRAGMA cache_size = 2;
|
|
|
|
CREATE TABLE t(x INTEGER PRIMARY KEY, y TEXT);
|
|
|
|
WITH s(i) AS (
|
|
|
|
VALUES(1) UNION ALL SELECT i+1 FROM s WHERE i<1900
|
|
|
|
)
|
|
|
|
INSERT INTO t(y) SELECT $big FROM s;
|
|
|
|
}
|
|
|
|
db close
|
|
|
|
|
|
|
|
testvfs tvfs -default 1
|
|
|
|
tvfs script vfs_callback
|
|
|
|
tvfs filter xRead
|
|
|
|
|
|
|
|
sqlite3 db test.db
|
|
|
|
proc vfs_callback {method file args} {
|
|
|
|
if {$file=="" && [info exists ::tmp_read_fail]} {
|
|
|
|
incr ::tmp_read_fail -1
|
|
|
|
if {$::tmp_read_fail<=0} {
|
|
|
|
return "SQLITE_IOERR"
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return "SQLITE_OK"
|
|
|
|
}
|
|
|
|
|
|
|
|
set FAULTSIM(tmpread) [list \
|
|
|
|
-injectstart tmpread_injectstart \
|
|
|
|
-injectstop tmpread_injectstop \
|
|
|
|
-injecterrlist {{1 {disk I/O error}}} \
|
|
|
|
]
|
|
|
|
proc tmpread_injectstart {iFail} {
|
|
|
|
set ::tmp_read_fail $iFail
|
|
|
|
}
|
|
|
|
proc tmpread_injectstop {} {
|
|
|
|
set ret [expr $::tmp_read_fail<=0]
|
|
|
|
unset -nocomplain ::tmp_read_fail
|
|
|
|
return $ret
|
|
|
|
}
|
|
|
|
|
2019-07-09 15:03:21 +03:00
|
|
|
set L [db eval {SELECT 0.0 FROM t}]
|
2019-03-30 20:07:23 +03:00
|
|
|
do_faultsim_test 9 -end 25 -faults tmpread -body {
|
|
|
|
execsql {
|
|
|
|
SELECT sum(y) OVER win FROM t
|
|
|
|
WINDOW win AS (
|
|
|
|
ORDER BY x ROWS BETWEEN UNBOUNDED PRECEDING AND 1800 FOLLOWING
|
|
|
|
)
|
|
|
|
}
|
|
|
|
} -test {
|
2019-07-09 15:03:21 +03:00
|
|
|
faultsim_test_result [list 0 $::L]
|
2019-03-30 20:07:23 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
catch {db close}
|
|
|
|
tvfs delete
|
|
|
|
|
2019-08-20 17:43:01 +03:00
|
|
|
reset_db
|
|
|
|
do_execsql_test 10.0 {
|
|
|
|
CREATE TABLE t1(a, b, c, d);
|
|
|
|
CREATE TABLE t2(a, b, c, d);
|
|
|
|
}
|
|
|
|
|
|
|
|
do_faultsim_test 1 -faults oom* -prep {
|
|
|
|
} -body {
|
|
|
|
execsql {
|
|
|
|
SELECT row_number() OVER win
|
|
|
|
FROM t1
|
|
|
|
WINDOW win AS (
|
|
|
|
ORDER BY (
|
|
|
|
SELECT percent_rank() OVER win2 FROM t2
|
|
|
|
WINDOW win2 AS (ORDER BY a)
|
|
|
|
)
|
|
|
|
)
|
|
|
|
}
|
|
|
|
} -test {
|
|
|
|
faultsim_test_result {0 {}}
|
|
|
|
}
|
|
|
|
|
2018-06-15 22:01:35 +03:00
|
|
|
finish_test
|