Documentation
¶
Index ¶
- Variables
- type GenericErrorQueryTest
- type QueryErrorTest
- type QueryPlanTest
- type QueryTest
- type QuickPrivilegeTest
- type ScriptTest
- type ScriptTestAssertion
- type ServerAuthenticationTest
- type ServerAuthenticationTestAssertion
- type TransactionTest
- type TypeWireTest
- type UserPrivilegeTest
- type UserPrivilegeTestAssertion
- type WriteQueryTest
Constants ¶
This section is empty.
Variables ¶
var BrokenQueries = []QueryTest{ { Query: "SELECT pk1, SUM(c1) FROM two_pk", Expected: []sql.Row{{0, 60.0}}, }, { Query: `SELECT pk, (SELECT max(pk) FROM one_pk WHERE pk < opk.pk) AS x FROM one_pk opk WHERE x > 0 ORDER BY x`, Expected: []sql.Row{ {2, 1}, {3, 2}, }, }, { Query: `SELECT pk, (SELECT max(pk) FROM one_pk WHERE pk < opk.pk) AS min, (SELECT min(pk) FROM one_pk WHERE pk > opk.pk) AS max FROM one_pk opk WHERE max > 1 ORDER BY max;`, Expected: []sql.Row{ {1, 0, 2}, {2, 1, 3}, }, }, { Query: `SELECT pk, (SELECT sum(c1) FROM two_pk WHERE c1 IN (SELECT c4 FROM two_pk WHERE c3 > opk.c5)) AS sum, (SELECT avg(c1) FROM two_pk WHERE pk2 IN (SELECT pk2 FROM two_pk WHERE c1 < opk.c2)) AS avg FROM one_pk opk ORDER BY pk`, Expected: []sql.Row{ {0, 60.0, nil}, {1, 50.0, 10.0}, {2, 30.0, 15.0}, {3, nil, 15.0}, }, }, { Query: `SELECT column_0, sum(column_1) FROM (values row(1,1), row(1,3), row(2,2), row(2,5), row(3,9)) a group by 1 having avg(column_1) > 2 order by 1`, Expected: []sql.Row{ {2, 7.0}, {3, 9.0}, }, }, { Query: `WITH t AS (SELECT 1) SELECT * FROM t UNION (WITH t AS (SELECT 2) SELECT * FROM t)`, Expected: []sql.Row{ {1}, {2}, }, }, { Query: "SELECT json_array() FROM dual;", }, { Query: "SELECT json_array_append() FROM dual;", }, { Query: "SELECT json_array_insert() FROM dual;", }, { Query: "SELECT json_contains() FROM dual;", }, { Query: "SELECT json_contains_path() FROM dual;", }, { Query: "SELECT json_depth() FROM dual;", }, { Query: "SELECT json_insert() FROM dual;", }, { Query: "SELECT json_keys() FROM dual;", }, { Query: "SELECT json_length() FROM dual;", }, { Query: "SELECT json_merge_patch() FROM dual;", }, { Query: "SELECT json_merge_preserve() FROM dual;", }, { Query: "SELECT json_object() FROM dual;", }, { Query: "SELECT json_overlaps() FROM dual;", }, { Query: "SELECT json_pretty() FROM dual;", }, { Query: "SELECT json_quote() FROM dual;", }, { Query: "SELECT json_remove() FROM dual;", }, { Query: "SELECT json_replace() FROM dual;", }, { Query: "SELECT json_schema_valid() FROM dual;", }, { Query: "SELECT json_schema_validation_report() FROM dual;", }, { Query: "SELECT json_set() FROM dual;", }, { Query: "SELECT json_search() FROM dual;", }, { Query: "SELECT json_storage_free() FROM dual;", }, { Query: "SELECT json_storage_size() FROM dual;", }, { Query: "SELECT json_type() FROM dual;", }, { Query: "SELECT json_table() FROM dual;", }, { Query: "SELECT json_valid() FROM dual;", }, { Query: "SELECT json_value() FROM dual;", }, { Query: `SELECT s as i, i as i from mytable order by i`, }, { Query: "SELECT i, I, s, S FROM mytable;", Expected: []sql.Row{ {1, 1, "first row", "first row"}, {2, 2, "second row", "second row"}, {3, 3, "third row", "third row"}, }, ExpectedColumns: sql.Schema{ { Name: "i", Type: sql.Int64, }, { Name: "I", Type: sql.Int64, }, { Name: "s", Type: sql.MustCreateStringWithDefaults(sqltypes.VarChar, 20), }, { Name: "S", Type: sql.MustCreateStringWithDefaults(sqltypes.VarChar, 20), }, }, }, { Query: "SELECT `i`, `I`, `s`, `S` FROM mytable;", Expected: []sql.Row{ {1, 1, "first row", "first row"}, {2, 2, "second row", "second row"}, {3, 3, "third row", "third row"}, }, ExpectedColumns: sql.Schema{ { Name: "i", Type: sql.Int64, }, { Name: "I", Type: sql.Int64, }, { Name: "s", Type: sql.MustCreateStringWithDefaults(sqltypes.VarChar, 20), }, { Name: "S", Type: sql.MustCreateStringWithDefaults(sqltypes.VarChar, 20), }, }, }, { Query: "SELECT `mytable`.`i`, `mytable`.`I`, `mytable`.`s`, `mytable`.`S` FROM mytable;", Expected: []sql.Row{ {1, 1, "first row", "first row"}, {2, 2, "second row", "second row"}, {3, 3, "third row", "third row"}, }, ExpectedColumns: sql.Schema{ { Name: "i", Type: sql.Int64, }, { Name: "I", Type: sql.Int64, }, { Name: "s", Type: sql.MustCreateStringWithDefaults(sqltypes.VarChar, 20), }, { Name: "S", Type: sql.MustCreateStringWithDefaults(sqltypes.VarChar, 20), }, }, }, { Query: `SELECT json_unquote(json_extract('{"hi":"there"}', '$.nope'))`, Expected: []sql.Row{{nil}}, }, { Query: "SELECT 1 FROM DUAL WHERE (1, null) in ((1, null))", Expected: []sql.Row{}, }, { Query: "SELECT 1 FROM DUAL WHERE (1, null) != (0, null)", Expected: []sql.Row{}, }, { Query: "SELECT 1 FROM DUAL WHERE (0, null) = (0, null)", Expected: []sql.Row{}, }, { Query: "SELECT 1 FROM DUAL WHERE ('0', 0) = (0, '0')", Expected: []sql.Row{{1}}, }, { Query: "SELECT 1 FROM DUAL WHERE (null, null) = (select null, null from dual)", Expected: []sql.Row{}, }, { Query: "SELECT c AS i_do_not_conflict, COUNT(*), MIN((SELECT COUNT(*) FROM (SELECT 1 AS d) b WHERE b.d = a.c)) FROM (SELECT 1 AS c) a GROUP BY i_do_not_conflict;", }, { Query: "SELECT c AS c, COUNT(*), MIN((SELECT COUNT(*) FROM (SELECT 1 AS d) b WHERE b.d = a.c)) FROM (SELECT 1 AS c) a GROUP BY a.c;", }, { Query: ` with recursive t1 (sub_part, part, quantity) as ( with recursive t2 (sub_part, part, quantity) as ( SELECT p2.sub_part, p2.part, p2.quantity FROM parts as p2 UNION SELECT p1.sub_part, p1.part, p1.quantity FROM parts as p1 JOIN t2 ON p1.sub_part = t2.sub_part WHERE p1.part = 'pie' and t2.part = 'crust' ) select * from t2 UNION SELECT t1.sub_part, t1.part, t1.quantity FROM t1 JOIN parts AS p ON p.part = p.part ) SELECT t1.sub_part, sum(t1.quantity) as total_quantity FROM t1 GROUP BY t1.sub_part;`, Expected: []sql.Row{ {"crust", float64(1)}, {"filling", float64(2)}, {"flour", float64(20)}, {"butter", float64(18)}, {"salt", float64(18)}, {"sugar", float64(7)}, {"fruit", float64(9)}, }, }, { Query: "select i, date_col from datetime_table", Expected: []sql.Row{{1, "2019-12-31"}}, }, }
Queries that are known to be broken in the engine.
var BrokenScriptTests = []ScriptTest{ { Name: "ALTER TABLE MODIFY column with multiple UNIQUE KEYS", SetUpScript: []string{ "CREATE table test (pk int primary key, uk1 int, uk2 int, unique(uk1, uk2))", "ALTER TABLE `test` MODIFY column uk1 int auto_increment", }, Assertions: []ScriptTestAssertion{ { Query: "describe test", Expected: []sql.Row{ {"pk", "int", "NO", "PRI", "", ""}, {"uk1", "int", "YES", "UNI", "", "auto_increment"}, {"uk1", "int", "YES", "UNI", "", "auto_increment"}, }, }, }, }, { Name: "ALTER TABLE MODIFY column with multiple KEYS", SetUpScript: []string{ "CREATE table test (pk int primary key, mk1 int, mk2 int, index(uk1, uk2))", "ALTER TABLE `test` MODIFY column mk1 int auto_increment", }, Assertions: []ScriptTestAssertion{ { Query: "describe test", Expected: []sql.Row{ {"pk", "int", "NO", "PRI", "", ""}, {"mk1", "int", "YES", "MUL", "", "auto_increment"}, {"mk1", "int", "YES", "MUL", "", "auto_increment"}, }, }, }, }, { Name: "Keyless Table with Unique Index", SetUpScript: []string{ "create table a (x int, val int unique)", }, Assertions: []ScriptTestAssertion{ { Query: "INSERT INTO a VALUES (1, 1)", Expected: []sql.Row{{sql.NewOkResult(1)}}, }, { Query: "INSERT INTO a VALUES (1, 1)", ExpectedErr: sql.ErrUniqueKeyViolation, }, }, }, { Name: "Multialter DDL with ADD/DROP Primary Key", SetUpScript: []string{ "CREATE TABLE t(pk int primary key, v1 int)", }, Assertions: []ScriptTestAssertion{ { Query: "ALTER TABLE t ADD COLUMN (v2 int), drop primary key, add primary key (v2)", Expected: []sql.Row{{sql.NewOkResult(0)}}, }, { Query: "DESCRIBE t", Expected: []sql.Row{ {"pk", "int", "NO", "", "", ""}, {"v1", "int", "YES", "", "", ""}, {"v2", "int", "NO", "PRI", "", ""}, }, }, { Query: "ALTER TABLE t ADD COLUMN (v3 int), drop primary key, add primary key (notacolumn)", ExpectedErr: sql.ErrKeyColumnDoesNotExist, }, { Query: "DESCRIBE t", Expected: []sql.Row{ {"pk", "int", "NO", "", "", ""}, {"v1", "int", "YES", "", "", ""}, {"v2", "int", "NO", "PRI", "", ""}, }, }, { Query: "ALTER TABLE t ADD column `v4` int NOT NULL, ADD column `v5` int NOT NULL, DROP COLUMN `v1`, ADD COLUMN `v6` int NOT NULL, DROP COLUMN `v2`, ADD COLUMN v7 int NOT NULL", Expected: []sql.Row{{sql.NewOkResult(0)}}, }, { Query: "DESCRIBE t", Expected: []sql.Row{ {"pk", "int", "NO", "", "", ""}, {"v4", "int", "NO", "", "", ""}, {"v5", "int", "NO", "", "", ""}, {"v6", "int", "NO", "", "", ""}, {"v7", "int", "NO", "", "", ""}, }, }, }, }, }
var BrokenTriggerQueries = []ScriptTest{ { Name: "update common table multiple times in single insert", SetUpScript: []string{ "create table mytable (id integer PRIMARY KEY DEFAULT 0, sometext text);", "create table sequence_table (max_id integer PRIMARY KEY);", "create trigger update_position_id before insert on mytable for each row begin set new.id = (select coalesce(max(max_id),1) from sequence_table); update sequence_table set max_id = max_id + 1; end;", "insert into sequence_table values (1);", }, Assertions: []ScriptTestAssertion{ { Query: "insert into mytable () values (), ();", Expected: []sql.Row{{sql.NewOkResult(2)}}, }, { Query: "select * from mytable order by id", Expected: []sql.Row{ {1, nil}, {2, nil}, {3, nil}, }, }, }, }, { Name: "insert into table multiple times", SetUpScript: []string{ "CREATE TABLE test(pk BIGINT PRIMARY KEY, v1 BIGINT);", "CREATE TABLE test2(pk BIGINT PRIMARY KEY, v1 BIGINT);", "INSERT INTO test VALUES (0,2),(1,3)", `CREATE TRIGGER tt BEFORE INSERT ON test FOR EACH ROW BEGIN insert into test2 values (new.pk * 3, new.v1); insert into test2 values (new.pk * 5, new.v1); END;`, "INSERT INTO test VALUES (2,4), (6,8);", }, Assertions: []ScriptTestAssertion{ { Query: "SELECT * FROM test ORDER BY 1", Expected: []sql.Row{ {0, 2}, {1, 3}, {2, -440}, }, }, { Query: "SELECT * FROM test2 ORDER BY 1", Expected: []sql.Row{ {2, -440}, }, }, }, }, { Name: "trigger after update, delete from other table", SetUpScript: []string{ "create table foo.a (x int primary key)", "create table foo.b (y int primary key)", "insert into foo.a values (0), (2), (4), (6), (8)", "insert into foo.b values (1), (3), (5), (7), (9)", "use foo", "create trigger insert_into_b after update on a for each row insert into b values (old.x + new.x + 1)", "use mydb", "update foo.a set x = x + 1 where x in (2,4)", }, Assertions: []ScriptTestAssertion{ { Query: "select x from foo.a order by 1", Expected: []sql.Row{ {0}, {3}, {5}, {6}, {8}, }, }, { Query: "select y from foo.b order by 1", Expected: []sql.Row{ {1}, {3}, {7}, }, }, }, }, { Name: "trigger before update, begin block with references to other table", SetUpScript: []string{ "CREATE TABLE a (i int primary key, j int)", "INSERT INTO a VALUES (0,1),(2,3),(4,5)", "CREATE TABLE b (x int)", "INSERT INTO b VALUES (1)", "CREATE TRIGGER trig BEFORE UPDATE ON a FOR EACH ROW BEGIN SET NEW.i = (SELECT x FROM b); SET NEW.j = OLD.j + NEW.j; UPDATE b SET x = x + 1; END;", "UPDATE a SET j = 10;", }, Assertions: []ScriptTestAssertion{ { Query: "SELECT * FROM a ORDER BY 1", Expected: []sql.Row{ {1, 11}, {2, 13}, {3, 15}, }, }, { Query: "SELECT * FROM b ORDER BY x", Expected: []sql.Row{ {4}, }, }, }, }, { Name: "trigger after inserts, use updated self reference", SetUpScript: []string{ "create table a (i int primary key, j int)", "create table b (x int primary key)", "insert into b values (1)", "create trigger trig after insert on a for each row begin update b set x = (select count(*) from a); end;", "insert into a values (1,0), (2,0), (3,0)", }, Assertions: []ScriptTestAssertion{ { Query: "select * from a order by i", Expected: []sql.Row{ {1, 0}, {2, 0}, {3, 0}, }, }, { Query: "select x from b", Expected: []sql.Row{ {3}, }, }, { Query: "insert into a values (4,0), (5,0)", Expected: []sql.Row{ {sql.OkResult{RowsAffected: 2}}, }, }, }, }, }
BrokenTriggerQueries contains trigger queries that should work but do not yet
var ColumnAliasQueries = []QueryTest{ { Query: `SELECT i AS cOl FROM mytable`, ExpectedColumns: sql.Schema{ { Name: "cOl", Type: sql.Int64, }, }, Expected: []sql.Row{ {int64(1)}, {int64(2)}, {int64(3)}, }, }, { Query: `SELECT i AS cOl, s as COL FROM mytable`, ExpectedColumns: sql.Schema{ { Name: "cOl", Type: sql.Int64, }, { Name: "COL", Type: sql.MustCreateStringWithDefaults(sqltypes.VarChar, 20), }, }, Expected: []sql.Row{ {int64(1), "first row"}, {int64(2), "second row"}, {int64(3), "third row"}, }, }, { Query: `SELECT i AS cOl, s as COL FROM mytable where cOl = 1`, ExpectedColumns: sql.Schema{ { Name: "cOl", Type: sql.Int64, }, { Name: "COL", Type: sql.MustCreateStringWithDefaults(sqltypes.VarChar, 20), }, }, Expected: []sql.Row{ {int64(1), "first row"}, }, }, { Query: `SELECT s as COL1, SUM(i) COL2 FROM mytable group by s order by cOL2`, ExpectedColumns: sql.Schema{ { Name: "COL1", Type: sql.MustCreateStringWithDefaults(sqltypes.VarChar, 20), }, { Name: "COL2", Type: sql.Float64, }, }, Expected: []sql.Row{ {"first row", float64(1)}, {"second row", float64(2)}, {"third row", float64(3)}, }, }, { Query: `SELECT s as COL1, SUM(i) COL2 FROM mytable group by col1 order by col2`, ExpectedColumns: sql.Schema{ { Name: "COL1", Type: sql.MustCreateStringWithDefaults(sqltypes.VarChar, 20), }, { Name: "COL2", Type: sql.Float64, }, }, Expected: []sql.Row{ {"first row", float64(1)}, {"second row", float64(2)}, {"third row", float64(3)}, }, }, { Query: `SELECT s as coL1, SUM(i) coL2 FROM mytable group by 1 order by 2`, ExpectedColumns: sql.Schema{ { Name: "coL1", Type: sql.MustCreateStringWithDefaults(sqltypes.VarChar, 20), }, { Name: "coL2", Type: sql.Float64, }, }, Expected: []sql.Row{ {"first row", float64(1)}, {"second row", float64(2)}, {"third row", float64(3)}, }, }, { Query: `SELECT s as Date, SUM(i) TimeStamp FROM mytable group by 1 order by 2`, ExpectedColumns: sql.Schema{ { Name: "Date", Type: sql.MustCreateStringWithDefaults(sqltypes.VarChar, 20), }, { Name: "TimeStamp", Type: sql.Float64, }, }, Expected: []sql.Row{ {"first row", float64(1)}, {"second row", float64(2)}, {"third row", float64(3)}, }, }, }
var ComplexIndexQueries = []QueryTest{}/* 701 elements not displayed */
var CreateCheckConstraintsScripts = []ScriptTest{ { Name: "Run SHOW CREATE TABLE with different types of check constraints", SetUpScript: []string{ "CREATE TABLE mytable1(pk int PRIMARY KEY, CONSTRAINT check1 CHECK (pk = 5))", "ALTER TABLE mytable1 ADD CONSTRAINT check11 CHECK (pk < 6)", "CREATE TABLE mytable2(pk int PRIMARY KEY, v int, CONSTRAINT check2 CHECK (v < 5))", "ALTER TABLE mytable2 ADD CONSTRAINT check12 CHECK (pk + v = 6)", "CREATE TABLE mytable3(pk int PRIMARY KEY, v int, CONSTRAINT check3 CHECK (pk > 2 AND v < 5))", "ALTER TABLE mytable3 ADD CONSTRAINT check13 CHECK (pk BETWEEN 2 AND 100)", "CREATE TABLE mytable4(pk int PRIMARY KEY, v int, CONSTRAINT check4 CHECK (pk > 2 AND v < 5 AND pk < 9))", "CREATE TABLE mytable5(pk int PRIMARY KEY, v int, CONSTRAINT check5 CHECK (pk > 2 OR (v < 5 AND pk < 9)))", "CREATE TABLE mytable6(pk int PRIMARY KEY, v int, CONSTRAINT check6 CHECK (NOT pk))", "CREATE TABLE mytable7(pk int PRIMARY KEY, v int, CONSTRAINT check7 CHECK (pk != v))", "CREATE TABLE mytable8(pk int PRIMARY KEY, v int, CONSTRAINT check8 CHECK (pk > 2 OR v < 5 OR pk < 10))", "CREATE TABLE mytable9(pk int PRIMARY KEY, v int, CONSTRAINT check9 CHECK ((pk + v) / 2 >= 1))", "CREATE TABLE mytable10(pk int PRIMARY KEY, v int, CONSTRAINT check10 CHECK (v < 5) NOT ENFORCED)", }, Assertions: []ScriptTestAssertion{ { Query: "SHOW CREATE TABLE mytable1", Expected: []sql.Row{ { "mytable1", "CREATE TABLE `mytable1` (\n `pk` int NOT NULL,\n" + " PRIMARY KEY (`pk`),\n" + " CONSTRAINT `check1` CHECK ((`pk` = 5)),\n" + " CONSTRAINT `check11` CHECK ((`pk` < 6))\n" + ") ENGINE=InnoDB DEFAULT CHARSET=utf8mb4 COLLATE=utf8mb4_0900_bin", }, }, }, { Query: "SHOW CREATE TABLE mytable2", Expected: []sql.Row{ { "mytable2", "CREATE TABLE `mytable2` (\n `pk` int NOT NULL,\n" + " `v` int,\n" + " PRIMARY KEY (`pk`),\n" + " CONSTRAINT `check2` CHECK ((`v` < 5)),\n" + " CONSTRAINT `check12` CHECK (((`pk` + `v`) = 6))\n" + ") ENGINE=InnoDB DEFAULT CHARSET=utf8mb4 COLLATE=utf8mb4_0900_bin", }, }, }, { Query: "SHOW CREATE TABLE mytable3", Expected: []sql.Row{ { "mytable3", "CREATE TABLE `mytable3` (\n `pk` int NOT NULL,\n" + " `v` int,\n" + " PRIMARY KEY (`pk`),\n" + " CONSTRAINT `check3` CHECK (((`pk` > 2) AND (`v` < 5))),\n" + " CONSTRAINT `check13` CHECK ((`pk` BETWEEN 2 AND 100))\n" + ") ENGINE=InnoDB DEFAULT CHARSET=utf8mb4 COLLATE=utf8mb4_0900_bin", }, }, }, { Query: "SHOW CREATE TABLE mytable4", Expected: []sql.Row{ { "mytable4", "CREATE TABLE `mytable4` (\n `pk` int NOT NULL,\n" + " `v` int,\n" + " PRIMARY KEY (`pk`),\n" + " CONSTRAINT `check4` CHECK ((((`pk` > 2) AND (`v` < 5)) AND (`pk` < 9)))\n" + ") ENGINE=InnoDB DEFAULT CHARSET=utf8mb4 COLLATE=utf8mb4_0900_bin", }, }, }, { Query: "SHOW CREATE TABLE mytable5", Expected: []sql.Row{ { "mytable5", "CREATE TABLE `mytable5` (\n `pk` int NOT NULL,\n" + " `v` int,\n" + " PRIMARY KEY (`pk`),\n" + " CONSTRAINT `check5` CHECK (((`pk` > 2) OR ((`v` < 5) AND (`pk` < 9))))\n" + ") ENGINE=InnoDB DEFAULT CHARSET=utf8mb4 COLLATE=utf8mb4_0900_bin", }, }, }, { Query: "SHOW CREATE TABLE mytable6", Expected: []sql.Row{ { "mytable6", "CREATE TABLE `mytable6` (\n `pk` int NOT NULL,\n" + " `v` int,\n" + " PRIMARY KEY (`pk`),\n" + " CONSTRAINT `check6` CHECK ((NOT(`pk`)))\n" + ") ENGINE=InnoDB DEFAULT CHARSET=utf8mb4 COLLATE=utf8mb4_0900_bin", }, }, }, { Query: "SHOW CREATE TABLE mytable7", Expected: []sql.Row{ { "mytable7", "CREATE TABLE `mytable7` (\n `pk` int NOT NULL,\n" + " `v` int,\n" + " PRIMARY KEY (`pk`),\n" + " CONSTRAINT `check7` CHECK ((NOT((`pk` = `v`))))\n" + ") ENGINE=InnoDB DEFAULT CHARSET=utf8mb4 COLLATE=utf8mb4_0900_bin", }, }, }, { Query: "SHOW CREATE TABLE mytable8", Expected: []sql.Row{ { "mytable8", "CREATE TABLE `mytable8` (\n `pk` int NOT NULL,\n" + " `v` int,\n" + " PRIMARY KEY (`pk`),\n" + " CONSTRAINT `check8` CHECK ((((`pk` > 2) OR (`v` < 5)) OR (`pk` < 10)))\n" + ") ENGINE=InnoDB DEFAULT CHARSET=utf8mb4 COLLATE=utf8mb4_0900_bin", }, }, }, { Query: "SHOW CREATE TABLE mytable9", Expected: []sql.Row{ { "mytable9", "CREATE TABLE `mytable9` (\n `pk` int NOT NULL,\n" + " `v` int,\n" + " PRIMARY KEY (`pk`),\n" + " CONSTRAINT `check9` CHECK ((((`pk` + `v`) / 2) >= 1))\n" + ") ENGINE=InnoDB DEFAULT CHARSET=utf8mb4 COLLATE=utf8mb4_0900_bin", }, }, }, { Query: "SHOW CREATE TABLE mytable10", Expected: []sql.Row{ { "mytable10", "CREATE TABLE `mytable10` (\n `pk` int NOT NULL,\n" + " `v` int,\n" + " PRIMARY KEY (`pk`),\n" + " CONSTRAINT `check10` CHECK ((`v` < 5)) /*!80016 NOT ENFORCED */\n" + ") ENGINE=InnoDB DEFAULT CHARSET=utf8mb4 COLLATE=utf8mb4_0900_bin", }, }, }, }, }, { Name: "Create a table with a check and validate that it appears in check_constraints and table_constraints", SetUpScript: []string{ "CREATE TABLE mytable (pk int primary key, test_score int, height int, CONSTRAINT mycheck CHECK (test_score >= 50), CONSTRAINT hcheck CHECK (height < 10), CONSTRAINT vcheck CHECK (height > 0))", }, Assertions: []ScriptTestAssertion{ { Query: "SELECT * from information_schema.check_constraints where constraint_name IN ('mycheck', 'hcheck') ORDER BY constraint_name", Expected: []sql.Row{ {"def", "mydb", "hcheck", "(height < 10)"}, {"def", "mydb", "mycheck", "(test_score >= 50)"}, }, }, { Query: "SELECT * FROM information_schema.table_constraints where table_name='mytable' ORDER BY constraint_type,constraint_name", Expected: []sql.Row{ {"def", "mydb", "hcheck", "mydb", "mytable", "CHECK", "YES"}, {"def", "mydb", "mycheck", "mydb", "mytable", "CHECK", "YES"}, {"def", "mydb", "vcheck", "mydb", "mytable", "CHECK", "YES"}, {"def", "mydb", "PRIMARY", "mydb", "mytable", "PRIMARY KEY", "YES"}, }, }, }, }, { Name: "multi column index, lower()", SetUpScript: []string{ "CREATE TABLE test (pk BIGINT PRIMARY KEY, v1 varchar(100), v2 varchar(100), INDEX (v1,v2));", "INSERT INTO test VALUES (1,'happy','birthday'), (2,'HAPPY','BIRTHDAY'), (3,'hello','sailor');", }, Assertions: []ScriptTestAssertion{ { Query: "SELECT pk FROM test where lower(v1) = 'happy' and lower(v2) = 'birthday' order by 1", Expected: []sql.Row{{1}, {2}}, }, }, }, { Name: "adding check constraint to a table that violates said constraint correctly throws an error", SetUpScript: []string{ "CREATE TABLE test (pk int)", "INSERT INTO test VALUES (1),(2),(300)", }, Assertions: []ScriptTestAssertion{ { Query: "ALTER TABLE test ADD CONSTRAINT bad_check CHECK (pk < 5)", ExpectedErr: plan.ErrCheckViolated, }, }, }, { Name: "duplicate indexes still returns correct results", SetUpScript: []string{ "CREATE TABLE test (i int)", "CREATE INDEX test_idx1 on test (i)", "CREATE INDEX test_idx2 on test (i)", "INSERT INTO test values (1), (2), (3)", }, Assertions: []ScriptTestAssertion{ { Query: "SELECT * FROM test ORDER BY i", Expected: []sql.Row{{1}, {2}, {3}}, }, { Query: "SELECT * FROM test where i = 2", Expected: []sql.Row{ {2}, }, }, }, }, }
var CreateTableQueries = []WriteQueryTest{ { WriteQuery: `CREATE TABLE t1 (a INTEGER, b TEXT, c DATE, d TIMESTAMP, e VARCHAR(20), f BLOB NOT NULL, b1 BOOL, b2 BOOLEAN NOT NULL, g DATETIME, h CHAR(40))`, ExpectedWriteResult: []sql.Row{{sql.NewOkResult(0)}}, SelectQuery: "SHOW CREATE TABLE t1", ExpectedSelect: []sql.Row{sql.Row{"t1", "CREATE TABLE `t1` (\n `a` int,\n `b` text,\n `c` date,\n `d` timestamp,\n `e` varchar(20),\n `f` blob NOT NULL,\n `b1` tinyint,\n `b2` tinyint NOT NULL,\n `g` datetime,\n `h` char(40)\n) ENGINE=InnoDB DEFAULT CHARSET=utf8mb4 COLLATE=utf8mb4_0900_bin"}}, }, { WriteQuery: `CREATE TABLE t1 (a INTEGER NOT NULL PRIMARY KEY, b VARCHAR(10) NOT NULL)`, ExpectedWriteResult: []sql.Row{{sql.NewOkResult(0)}}, SelectQuery: "SHOW CREATE TABLE t1", ExpectedSelect: []sql.Row{sql.Row{"t1", "CREATE TABLE `t1` (\n `a` int NOT NULL,\n `b` varchar(10) NOT NULL,\n PRIMARY KEY (`a`)\n) ENGINE=InnoDB DEFAULT CHARSET=utf8mb4 COLLATE=utf8mb4_0900_bin"}}, }, { WriteQuery: `CREATE TABLE t1 (a INTEGER NOT NULL, b TEXT NOT NULL, c bool, primary key (a,b))`, ExpectedWriteResult: []sql.Row{{sql.NewOkResult(0)}}, SelectQuery: "SHOW CREATE TABLE t1", ExpectedSelect: []sql.Row{sql.Row{"t1", "CREATE TABLE `t1` (\n `a` int NOT NULL,\n `b` text NOT NULL,\n `c` tinyint,\n PRIMARY KEY (`a`,`b`)\n) ENGINE=InnoDB DEFAULT CHARSET=utf8mb4 COLLATE=utf8mb4_0900_bin"}}, }, { WriteQuery: `CREATE TABLE t1(a INTEGER NOT NULL, b TEXT NOT NULL, c bool, primary key (a,b))`, ExpectedWriteResult: []sql.Row{{sql.NewOkResult(0)}}, SelectQuery: "SHOW CREATE TABLE t1", ExpectedSelect: []sql.Row{sql.Row{"t1", "CREATE TABLE `t1` (\n `a` int NOT NULL,\n `b` text NOT NULL,\n `c` tinyint,\n PRIMARY KEY (`a`,`b`)\n) ENGINE=InnoDB DEFAULT CHARSET=utf8mb4 COLLATE=utf8mb4_0900_bin"}}, }, { WriteQuery: `CREATE TABLE t1 (a INTEGER, b TEXT NOT NULL COMMENT 'comment', c bool, primary key (a))`, ExpectedWriteResult: []sql.Row{{sql.NewOkResult(0)}}, SelectQuery: "SHOW CREATE TABLE t1", ExpectedSelect: []sql.Row{sql.Row{"t1", "CREATE TABLE `t1` (\n `a` int NOT NULL,\n `b` text NOT NULL COMMENT 'comment',\n `c` tinyint,\n PRIMARY KEY (`a`)\n) ENGINE=InnoDB DEFAULT CHARSET=utf8mb4 COLLATE=utf8mb4_0900_bin"}}, }, { WriteQuery: `CREATE TABLE t1 (a INTEGER, create_time timestamp(6) NOT NULL DEFAULT NOW(6), primary key (a))`, ExpectedWriteResult: []sql.Row{{sql.NewOkResult(0)}}, SelectQuery: "SHOW CREATE TABLE t1", ExpectedSelect: []sql.Row{sql.Row{"t1", "CREATE TABLE `t1` (\n `a` int NOT NULL,\n `create_time` timestamp NOT NULL DEFAULT (NOW(6)),\n PRIMARY KEY (`a`)\n) ENGINE=InnoDB DEFAULT CHARSET=utf8mb4 COLLATE=utf8mb4_0900_bin"}}, }, { WriteQuery: `CREATE TABLE t1 LIKE mytable`, ExpectedWriteResult: []sql.Row{{sql.NewOkResult(0)}}, SelectQuery: "SHOW CREATE TABLE t1", ExpectedSelect: []sql.Row{sql.Row{"t1", "CREATE TABLE `t1` (\n `i` bigint NOT NULL,\n `s` varchar(20) NOT NULL COMMENT 'column s',\n PRIMARY KEY (`i`),\n KEY `idx_si` (`s`,`i`),\n KEY `mytable_i_s` (`i`,`s`),\n UNIQUE KEY `mytable_s` (`s`)\n) ENGINE=InnoDB DEFAULT CHARSET=utf8mb4 COLLATE=utf8mb4_0900_bin"}}, }, { WriteQuery: `CREATE TABLE t1 ( pk bigint primary key, v1 bigint default (2) comment 'hi there', index idx_v1 (v1) comment 'index here' )`, ExpectedWriteResult: []sql.Row{{sql.NewOkResult(0)}}, SelectQuery: "SHOW CREATE TABLE t1", ExpectedSelect: []sql.Row{sql.Row{"t1", "CREATE TABLE `t1` (\n `pk` bigint NOT NULL,\n `v1` bigint DEFAULT (2) COMMENT 'hi there',\n PRIMARY KEY (`pk`),\n KEY `idx_v1` (`v1`) COMMENT 'index here'\n) ENGINE=InnoDB DEFAULT CHARSET=utf8mb4 COLLATE=utf8mb4_0900_bin"}}, }, { WriteQuery: `create table t1 like foo.other_table`, ExpectedWriteResult: []sql.Row{{sql.NewOkResult(0)}}, SelectQuery: "SHOW CREATE TABLE t1", ExpectedSelect: []sql.Row{sql.Row{"t1", "CREATE TABLE `t1` (\n `text` text NOT NULL,\n `number` mediumint,\n PRIMARY KEY (`text`)\n) ENGINE=InnoDB DEFAULT CHARSET=utf8mb4 COLLATE=utf8mb4_0900_bin"}}, }, { WriteQuery: `CREATE TABLE t1 (a INTEGER NOT NULL PRIMARY KEY, b VARCHAR(10) UNIQUE)`, ExpectedWriteResult: []sql.Row{{sql.NewOkResult(0)}}, SelectQuery: "SHOW CREATE TABLE t1", ExpectedSelect: []sql.Row{sql.Row{"t1", "CREATE TABLE `t1` (\n `a` int NOT NULL,\n `b` varchar(10),\n PRIMARY KEY (`a`),\n UNIQUE KEY `b` (`b`)\n) ENGINE=InnoDB DEFAULT CHARSET=utf8mb4 COLLATE=utf8mb4_0900_bin"}}, }, { WriteQuery: `CREATE TABLE t1 (a INTEGER NOT NULL PRIMARY KEY, b VARCHAR(10) UNIQUE KEY)`, ExpectedWriteResult: []sql.Row{{sql.NewOkResult(0)}}, SelectQuery: "SHOW CREATE TABLE t1", ExpectedSelect: []sql.Row{sql.Row{"t1", "CREATE TABLE `t1` (\n `a` int NOT NULL,\n `b` varchar(10),\n PRIMARY KEY (`a`),\n UNIQUE KEY `b` (`b`)\n) ENGINE=InnoDB DEFAULT CHARSET=utf8mb4 COLLATE=utf8mb4_0900_bin"}}, }, { WriteQuery: `CREATE TABLE t1 SELECT * from mytable`, ExpectedWriteResult: []sql.Row{{sql.NewOkResult(3)}}, SelectQuery: "SHOW CREATE TABLE t1", ExpectedSelect: []sql.Row{sql.Row{"t1", "CREATE TABLE `t1` (\n `i` bigint NOT NULL,\n `s` varchar(20) NOT NULL COMMENT 'column s',\n PRIMARY KEY (`i`)\n) ENGINE=InnoDB DEFAULT CHARSET=utf8mb4 COLLATE=utf8mb4_0900_bin"}}, }, { WriteQuery: `CREATE TABLE mydb.t1 (a INTEGER NOT NULL PRIMARY KEY, b VARCHAR(10) NOT NULL)`, ExpectedWriteResult: []sql.Row{{sql.NewOkResult(0)}}, SelectQuery: "SHOW CREATE TABLE mydb.t1", ExpectedSelect: []sql.Row{sql.Row{"t1", "CREATE TABLE `t1` (\n `a` int NOT NULL,\n `b` varchar(10) NOT NULL,\n PRIMARY KEY (`a`)\n) ENGINE=InnoDB DEFAULT CHARSET=utf8mb4 COLLATE=utf8mb4_0900_bin"}}, }, { WriteQuery: `CREATE TABLE t1 (i int primary key, j int auto_increment unique)`, ExpectedWriteResult: []sql.Row{{sql.NewOkResult(0)}}, SelectQuery: "SHOW CREATE TABLE t1", ExpectedSelect: []sql.Row{sql.Row{"t1", "CREATE TABLE `t1` (\n `i` int NOT NULL,\n `j` int AUTO_INCREMENT,\n PRIMARY KEY (`i`),\n UNIQUE KEY `j` (`j`)\n) ENGINE=InnoDB DEFAULT CHARSET=utf8mb4 COLLATE=utf8mb4_0900_bin"}}, }, { WriteQuery: `CREATE TABLE t1 (i int primary key, j int auto_increment, index (j))`, ExpectedWriteResult: []sql.Row{{sql.NewOkResult(0)}}, SelectQuery: "SHOW CREATE TABLE t1", ExpectedSelect: []sql.Row{sql.Row{"t1", "CREATE TABLE `t1` (\n `i` int NOT NULL,\n `j` int AUTO_INCREMENT,\n PRIMARY KEY (`i`),\n KEY `j` (`j`)\n) ENGINE=InnoDB DEFAULT CHARSET=utf8mb4 COLLATE=utf8mb4_0900_bin"}}, }, { WriteQuery: `CREATE TABLE t1 (i int primary key, j int auto_increment, k int, unique(j,k))`, ExpectedWriteResult: []sql.Row{{sql.NewOkResult(0)}}, SelectQuery: "SHOW CREATE TABLE t1", ExpectedSelect: []sql.Row{sql.Row{"t1", "CREATE TABLE `t1` (\n `i` int NOT NULL,\n `j` int AUTO_INCREMENT,\n `k` int,\n PRIMARY KEY (`i`),\n UNIQUE KEY `jk` (`j`,`k`)\n) ENGINE=InnoDB DEFAULT CHARSET=utf8mb4 COLLATE=utf8mb4_0900_bin"}}, }, { WriteQuery: `CREATE TABLE t1 (i int primary key, j int auto_increment, k int, index (j,k))`, ExpectedWriteResult: []sql.Row{{sql.NewOkResult(0)}}, SelectQuery: "SHOW CREATE TABLE t1", ExpectedSelect: []sql.Row{sql.Row{"t1", "CREATE TABLE `t1` (\n `i` int NOT NULL,\n `j` int AUTO_INCREMENT,\n `k` int,\n PRIMARY KEY (`i`),\n KEY `jk` (`j`,`k`)\n) ENGINE=InnoDB DEFAULT CHARSET=utf8mb4 COLLATE=utf8mb4_0900_bin"}}, }, { WriteQuery: `CREATE TABLE t1 ( pk int NOT NULL, col1 blob DEFAULT (_utf8mb4'abc'), col2 json DEFAULT (json_object(_utf8mb4'a',1)), col3 text DEFAULT (_utf8mb4'abc'), PRIMARY KEY (pk) )`, ExpectedWriteResult: []sql.Row{{sql.NewOkResult(0)}}, SelectQuery: "SHOW CREATE TABLE t1", ExpectedSelect: []sql.Row{sql.Row{"t1", "CREATE TABLE `t1` (\n `pk` int NOT NULL,\n `col1` blob DEFAULT (\"abc\"),\n `col2` json DEFAULT (JSON_OBJECT(\"a\", 1)),\n `col3` text DEFAULT (\"abc\"),\n PRIMARY KEY (`pk`)\n) ENGINE=InnoDB DEFAULT CHARSET=utf8mb4 COLLATE=utf8mb4_0900_bin"}}, }, }
var DateParseQueries = []QueryTest{ { Query: "SELECT STR_TO_DATE('Jan 3, 2000', '%b %e, %Y')", Expected: []sql.Row{{time.Date(2000, time.January, 3, 0, 0, 0, 0, time.Local)}}, }, { Query: "SELECT STR_TO_DATE('May 3, 10:23:00 PM 2000', '%b %e, %H:%i:%s %p %Y')", Expected: []sql.Row{{time.Date(2000, time.May, 3, 22, 23, 0, 0, time.Local)}}, }, { Query: "SELECT STR_TO_DATE('01/02/99 314', '%m/%e/%y %f')", Expected: []sql.Row{{time.Date(1999, time.January, 2, 0, 0, 0, 314000, time.Local)}}, }, { Query: "SELECT STR_TO_DATE('01/02/99 05:14:12 PM', '%m/%e/%y %r')", Expected: []sql.Row{{time.Date(1999, time.January, 2, 17, 14, 12, 0, time.Local)}}, }, { Query: "SELECT STR_TO_DATE('invalid', 'notvalid')", Expected: []sql.Row{{sql.Null}}, }, }
var DeleteErrorTests = []GenericErrorQueryTest{
{
Name: "invalid table",
Query: "DELETE FROM invalidtable WHERE x < 1;",
},
{
Name: "invalid column",
Query: "DELETE FROM mytable WHERE z = 'dne';",
},
{
Name: "missing binding",
Query: "DELETE FROM mytable WHERE i = ?;",
},
{
Name: "negative limit",
Query: "DELETE FROM mytable LIMIT -1;",
},
{
Name: "negative offset",
Query: "DELETE FROM mytable LIMIT 1 OFFSET -1;",
},
{
Name: "missing keyword from",
Query: "DELETE mytable WHERE id = 1;",
},
{
Name: "targets join",
Query: "DELETE FROM mytable one, mytable two WHERE id = 1;",
},
{
Name: "targets subquery alias",
Query: "DELETE FROM (SELECT * FROM mytable) mytable WHERE id = 1;",
},
}
var DeleteTests = []WriteQueryTest{ { WriteQuery: "DELETE FROM mytable;", ExpectedWriteResult: []sql.Row{{sql.NewOkResult(3)}}, SelectQuery: "SELECT * FROM mytable;", ExpectedSelect: nil, }, { WriteQuery: "DELETE FROM mytable WHERE i = 2;", ExpectedWriteResult: []sql.Row{{sql.NewOkResult(1)}}, SelectQuery: "SELECT * FROM mytable;", ExpectedSelect: []sql.Row{{int64(1), "first row"}, {int64(3), "third row"}}, }, { WriteQuery: "DELETE FROM mytable WHERE I = 2;", ExpectedWriteResult: []sql.Row{{sql.NewOkResult(1)}}, SelectQuery: "SELECT * FROM mytable;", ExpectedSelect: []sql.Row{{int64(1), "first row"}, {int64(3), "third row"}}, }, { WriteQuery: "DELETE FROM mytable WHERE i < 3;", ExpectedWriteResult: []sql.Row{{sql.NewOkResult(2)}}, SelectQuery: "SELECT * FROM mytable;", ExpectedSelect: []sql.Row{{int64(3), "third row"}}, }, { WriteQuery: "DELETE FROM mytable WHERE i > 1;", ExpectedWriteResult: []sql.Row{{sql.NewOkResult(2)}}, SelectQuery: "SELECT * FROM mytable;", ExpectedSelect: []sql.Row{{int64(1), "first row"}}, }, { WriteQuery: "DELETE FROM mytable WHERE i <= 2;", ExpectedWriteResult: []sql.Row{{sql.NewOkResult(2)}}, SelectQuery: "SELECT * FROM mytable;", ExpectedSelect: []sql.Row{{int64(3), "third row"}}, }, { WriteQuery: "DELETE FROM mytable WHERE i >= 2;", ExpectedWriteResult: []sql.Row{{sql.NewOkResult(2)}}, SelectQuery: "SELECT * FROM mytable;", ExpectedSelect: []sql.Row{{int64(1), "first row"}}, }, { WriteQuery: "DELETE FROM mytable WHERE s = 'first row';", ExpectedWriteResult: []sql.Row{{sql.NewOkResult(1)}}, SelectQuery: "SELECT * FROM mytable;", ExpectedSelect: []sql.Row{{int64(2), "second row"}, {int64(3), "third row"}}, }, { WriteQuery: "DELETE FROM mytable WHERE s <> 'dne';", ExpectedWriteResult: []sql.Row{{sql.NewOkResult(3)}}, SelectQuery: "SELECT * FROM mytable;", ExpectedSelect: nil, }, { WriteQuery: "DELETE FROM mytable WHERE i in (2,3);", ExpectedWriteResult: []sql.Row{{sql.NewOkResult(2)}}, SelectQuery: "SELECT * FROM mytable;", ExpectedSelect: []sql.Row{{int64(1), "first row"}}, }, { WriteQuery: "DELETE FROM mytable WHERE s LIKE '%row';", ExpectedWriteResult: []sql.Row{{sql.NewOkResult(3)}}, SelectQuery: "SELECT * FROM mytable;", ExpectedSelect: nil, }, { WriteQuery: "DELETE FROM mytable WHERE s = 'dne';", ExpectedWriteResult: []sql.Row{{sql.NewOkResult(0)}}, SelectQuery: "SELECT * FROM mytable;", ExpectedSelect: []sql.Row{{int64(1), "first row"}, {int64(2), "second row"}, {int64(3), "third row"}}, }, { WriteQuery: "DELETE FROM mytable WHERE i = 'invalid';", ExpectedWriteResult: []sql.Row{{sql.NewOkResult(0)}}, SelectQuery: "SELECT * FROM mytable;", ExpectedSelect: []sql.Row{{int64(1), "first row"}, {int64(2), "second row"}, {int64(3), "third row"}}, }, { WriteQuery: "DELETE FROM mytable ORDER BY i ASC LIMIT 2;", ExpectedWriteResult: []sql.Row{{sql.NewOkResult(2)}}, SelectQuery: "SELECT * FROM mytable;", ExpectedSelect: []sql.Row{{int64(3), "third row"}}, }, { WriteQuery: "DELETE FROM mytable ORDER BY i DESC LIMIT 1;", ExpectedWriteResult: []sql.Row{{sql.NewOkResult(1)}}, SelectQuery: "SELECT * FROM mytable;", ExpectedSelect: []sql.Row{{int64(1), "first row"}, {int64(2), "second row"}}, }, { WriteQuery: "DELETE FROM mytable ORDER BY i DESC LIMIT 1 OFFSET 1;", ExpectedWriteResult: []sql.Row{{sql.NewOkResult(1)}}, SelectQuery: "SELECT * FROM mytable;", ExpectedSelect: []sql.Row{{int64(1), "first row"}, {int64(3), "third row"}}, }, { WriteQuery: "DELETE FROM mytable WHERE (i,s) = (1, 'first row');", ExpectedWriteResult: []sql.Row{{sql.NewOkResult(1)}}, SelectQuery: "SELECT * FROM mytable;", ExpectedSelect: []sql.Row{{int64(2), "second row"}, {int64(3), "third row"}}, }, { WriteQuery: `DELETE FROM tabletest where 's' = 'something'`, ExpectedWriteResult: []sql.Row{{sql.OkResult{RowsAffected: 0}}}, SelectQuery: "SELECT * FROM mytable;", ExpectedSelect: []sql.Row{{int64(1), "first row"}, {int64(2), "second row"}, {int64(3), "third row"}}, }, }
var ErrorQueries = []QueryErrorTest{ { Query: "select * from dual where foo() and true;", ExpectedErr: sql.ErrFunctionNotFound, }, { Query: "select * from mytable where (i = 1, i = 0 or i = 2) and (i > -1)", ExpectedErr: sql.ErrInvalidOperandColumns, }, { Query: "select * from mytable where (i = 1, i = 0 or i = 2) or (i > -1)", ExpectedErr: sql.ErrInvalidOperandColumns, }, { Query: "select * from mytable where ((i = 1, i = 0 or i = 2) or (i > -1)) and (i < 6)", ExpectedErr: sql.ErrInvalidOperandColumns, }, { Query: "select * from mytable where ((i = 1, i = 0 or i = 2) is true or (i > -1)) and (i < 6)", ExpectedErr: sql.ErrInvalidOperandColumns, }, { Query: "select foo.i from mytable as a", ExpectedErr: sql.ErrTableNotFound, }, { Query: "select foo.i from mytable", ExpectedErr: sql.ErrTableNotFound, }, { Query: "select foo.* from mytable", ExpectedErr: sql.ErrTableNotFound, }, { Query: "select foo.* from mytable as a", ExpectedErr: sql.ErrTableNotFound, }, { Query: "select x from mytable", ExpectedErr: sql.ErrColumnNotFound, }, { Query: "select mytable.x from mytable", ExpectedErr: sql.ErrTableColumnNotFound, }, { Query: "select a.x from mytable as a", ExpectedErr: sql.ErrTableColumnNotFound, }, { Query: "select a from notable", ExpectedErr: sql.ErrTableNotFound, }, { Query: "select myTable.i from mytable as mt", ExpectedErr: sql.ErrTableNotFound, }, { Query: "select myTable.* from mytable as mt", ExpectedErr: sql.ErrTableNotFound, }, { Query: "SELECT one_pk.c5,pk1,pk2 FROM one_pk opk JOIN two_pk tpk ON one_pk.pk=two_pk.pk1 ORDER BY 1,2,3", ExpectedErr: sql.ErrTableNotFound, }, { Query: "SELECT pk,pk1,pk2 FROM one_pk opk JOIN two_pk tpk ON one_pk.pk=two_pk.pk1 AND opk.pk=tpk.pk2 ORDER BY 1,2,3", ExpectedErr: sql.ErrTableNotFound, }, { Query: "SELECT t.i, myview1.s FROM myview AS t ORDER BY i", ExpectedErr: sql.ErrTableNotFound, }, { Query: "SELECT * FROM mytable AS t, othertable as t", ExpectedErr: sql.ErrDuplicateAliasOrTable, }, { Query: "SELECT * FROM mytable AS t UNION SELECT * FROM mytable AS t, othertable AS t", ExpectedErr: sql.ErrDuplicateAliasOrTable, }, { Query: "SELECT * FROM mytable AS OTHERTABLE, othertable", ExpectedErr: sql.ErrDuplicateAliasOrTable, }, { Query: `SELECT * FROM mytable WHERE s REGEXP("*main.go")`, ExpectedErr: expression.ErrInvalidRegexp, }, { Query: `SELECT SUBSTRING(s, 1, 10) AS sub_s, SUBSTRING(SUB_S, 2, 3) AS sub_sub_s FROM mytable`, ExpectedErr: sql.ErrMisusedAlias, }, { Query: "SELECT pk, (SELECT max(pk) FROM one_pk b WHERE b.pk <= one_pk.pk) FROM one_pk opk ORDER BY 1", ExpectedErr: sql.ErrTableNotFound, }, { Query: "SELECT pk, (SELECT max(pk) FROM one_pk WHERE b.pk <= one_pk.pk) FROM one_pk opk ORDER BY 1", ExpectedErr: sql.ErrTableNotFound, }, { Query: "SELECT pk, (SELECT max(pk) FROM one_pk WHERE b.pk <= one_pk.pk) FROM one_pk opk ORDER BY 1", ExpectedErr: sql.ErrTableNotFound, }, { Query: "SELECT pk, (SELECT max(pk) FROM two_pk WHERE pk <= one_pk.pk3) FROM one_pk ORDER BY 1", ExpectedErr: sql.ErrTableColumnNotFound, }, { Query: "SELECT pk, (SELECT max(pk) FROM dne WHERE pk <= one_pk.pk3) FROM one_pk ORDER BY 1", ExpectedErr: sql.ErrTableNotFound, }, { Query: "SELECT pk, (SELECT max(pk) FROM two_pk WHERE pk <= c6) FROM one_pk ORDER BY 1", ExpectedErr: sql.ErrColumnNotFound, }, { Query: "SELECT i FROM myhistorytable AS OF abc", ExpectedErr: sql.ErrInvalidAsOfExpression, }, { Query: "SELECT i FROM myhistorytable AS OF MAX(abc)", ExpectedErr: sql.ErrInvalidAsOfExpression, }, { Query: "SELECT pk FROM one_pk WHERE pk > ?", ExpectedErr: sql.ErrUnboundPreparedStatementVariable, }, { Query: "SELECT pk FROM one_pk WHERE pk > :pk", ExpectedErr: sql.ErrUnboundPreparedStatementVariable, }, { Query: "with cte1 as (SELECT c3 FROM one_pk WHERE c4 < opk.c2 ORDER BY 1 DESC LIMIT 1) SELECT pk, (select c3 from cte1) FROM one_pk opk ORDER BY 1", ExpectedErr: sql.ErrTableNotFound, }, { Query: `WITH mt1 (x,y) as (select i,s FROM mytable) SELECT mt1.i, mt1.s FROM mt1`, ExpectedErr: sql.ErrTableColumnNotFound, }, { Query: `WITH mt1 (x,y) as (select i,s FROM mytable) SELECT i, s FROM mt1`, ExpectedErr: sql.ErrColumnNotFound, }, { Query: `WITH mt1 (x,y,z) as (select i,s FROM mytable) SELECT i, s FROM mt1`, ExpectedErr: sql.ErrColumnCountMismatch, }, { Query: `WITH mt1 as (select i,s FROM mytable), mt2 as (select i+1, concat(s, '!') from mytable) SELECT mt1.i, mt2.s FROM mt1 join mt2 on mt1.i = mt2.i;`, ExpectedErr: sql.ErrTableColumnNotFound, }, { Query: `SHOW TABLE STATUS FROM baddb`, ExpectedErr: sql.ErrDatabaseNotFound, }, { Query: `SELECT s as i, i as i from mytable order by 1`, ExpectedErr: sql.ErrAmbiguousColumnInOrderBy, }, { Query: `SELECT pk as pk, nt.i as i, nt2.i as i FROM one_pk RIGHT JOIN niltable nt ON pk=nt.i RIGHT JOIN niltable nt2 ON pk=nt2.i - 1 ORDER BY 3`, ExpectedErr: sql.ErrAmbiguousColumnInOrderBy, }, { Query: "SELECT C FROM (select i,s FROM mytable) mt (a,b) order by a desc;", ExpectedErr: sql.ErrColumnNotFound, }, { Query: "SELECT i FROM (select i,s FROM mytable) mt (a,b) order by a desc;", ExpectedErr: sql.ErrColumnNotFound, }, { Query: "SELECT mt.i FROM (select i,s FROM mytable) mt (a,b) order by a desc;", ExpectedErr: sql.ErrTableColumnNotFound, }, { Query: "SELECT a FROM (select i,s FROM mytable) mt (a) order by a desc;", ExpectedErr: sql.ErrColumnCountMismatch, }, { Query: "SELECT a FROM (select i,s FROM mytable) mt (a,b,c) order by a desc;", ExpectedErr: sql.ErrColumnCountMismatch, }, { Query: `SELECT name FROM specialtable t WHERE t.name LIKE '$%' ESCAPE 'abc'`, ExpectedErr: sql.ErrInvalidArgument, }, { Query: `SELECT name FROM specialtable t WHERE t.name LIKE '$%' ESCAPE '$$'`, ExpectedErr: sql.ErrInvalidArgument, }, { Query: `SELECT JSON_OBJECT("a","b","c") FROM dual`, ExpectedErr: sql.ErrInvalidArgumentNumber, }, { Query: `select JSON_EXTRACT('{"id":"abc"}', '$.id')-1;`, ExpectedErrStr: `error: 'abc' is not a valid value for 'DOUBLE'`, }, { Query: `select JSON_EXTRACT('{"id":{"a": "abc"}}', '$.id')-1;`, ExpectedErrStr: `error: 'map[string]interface {}' is not a valid value type for 'DOUBLE'`, }, { Query: `alter table mytable add primary key (s)`, ExpectedErr: sql.ErrMultiplePrimaryKeysDefined, }, { Query: "SELECT SUM(i), i FROM mytable GROUP BY i ORDER BY 1+SUM(i) ASC", ExpectedErr: analyzer.ErrAggregationUnsupported, }, { Query: "SELECT SUM(i) as sum, i FROM mytable GROUP BY i ORDER BY 1+SUM(i) ASC", ExpectedErr: analyzer.ErrAggregationUnsupported, }, { Query: "select ((1, 2)) from dual", ExpectedErr: sql.ErrInvalidOperandColumns, }, { Query: "select (select 1, 2 from dual) from dual", ExpectedErr: sql.ErrInvalidOperandColumns, }, { Query: "select concat((1, 2)) from dual", ExpectedErr: sql.ErrInvalidOperandColumns, }, { Query: "select (1, 2) = (1) from dual", ExpectedErr: sql.ErrInvalidOperandColumns, }, { Query: "select (1) in (select 1, 2 from dual) from dual", ExpectedErr: sql.ErrInvalidOperandColumns, }, { Query: "select (1, 2) in (select 1, 2, 3 from dual) from dual", ExpectedErr: sql.ErrInvalidOperandColumns, }, { Query: "select (select 1 from dual) in ((1, 2)) from dual", ExpectedErr: sql.ErrInvalidOperandColumns, }, { Query: "select (((1,2),3)) = (((1,2))) from dual", ExpectedErr: sql.ErrInvalidOperandColumns, }, { Query: "select (((1,2),3)) = (((1),2)) from dual", ExpectedErr: sql.ErrInvalidOperandColumns, }, { Query: "select (((1,2),3)) = (((1))) from dual", ExpectedErr: sql.ErrInvalidOperandColumns, }, { Query: "select (((1,2),3)) = (((1,2),3),(4,5)) from dual", ExpectedErr: sql.ErrInvalidOperandColumns, }, { Query: "select ((4,5),((1,2),3)) = ((1,2),(4,5)) from dual", ExpectedErr: sql.ErrInvalidOperandColumns, }, { Query: "SELECT (2, 2)=1 FROM dual where exists (SELECT 1 FROM dual)", ExpectedErr: sql.ErrInvalidOperandColumns, }, { Query: `SELECT pk, (SELECT concat(pk, pk) FROM one_pk WHERE pk < opk.pk ORDER BY 1 DESC LIMIT 1) as strpk FROM one_pk opk where strpk > "0" ORDER BY 2`, ExpectedErr: sql.ErrColumnNotFound, }, { Query: `CREATE TABLE test (pk int, primary key(pk, noexist))`, ExpectedErr: sql.ErrUnknownIndexColumn, }, { Query: `CREATE TABLE test (pk int auto_increment, pk2 int auto_increment, primary key (pk))`, ExpectedErr: sql.ErrInvalidAutoIncCols, }, { Query: `CREATE TABLE test (pk int auto_increment)`, ExpectedErr: sql.ErrInvalidAutoIncCols, }, { Query: `CREATE TABLE test (pk int primary key auto_increment default 100, col int)`, ExpectedErr: sql.ErrInvalidAutoIncCols, }, { Query: "with recursive t (n) as (select (1) from dual union all select n from t where n < 2) select sum(n) from t", ExpectedErr: sql.ErrCteRecursionLimitExceeded, }, { Query: "with recursive t (n) as (select (1) from dual union all select n + 1 from t where n < 1002) select sum(n) from t", ExpectedErr: sql.ErrCteRecursionLimitExceeded, }, { Query: `alter table a add fulltext index idx (id)`, ExpectedErr: sql.ErrUnsupportedFeature, }, { Query: `CREATE TABLE test (pk int primary key, body text, FULLTEXT KEY idx_body (body))`, ExpectedErr: sql.ErrUnsupportedFeature, }, { Query: `CREATE FULLTEXT INDEX idx ON opening_lines(opening_line)`, ExpectedErr: sql.ErrUnsupportedFeature, }, { Query: `SELECT * FROM datetime_table where date_col >= 'not a valid date'`, ExpectedErr: sql.ErrConvertingToTime, }, { Query: `SELECT * FROM datetime_table where datetime_col >= 'not a valid datetime'`, ExpectedErr: sql.ErrConvertingToTime, }, { Query: `SELECT t1.* FROM mytable as t1, mytable as t2, mytable as t3, mytable as t4, mytable as t5, mytable as t6, mytable as t7, mytable as t8, mytable as t9, mytable as t10, mytable as t11, mytable as t12, mytable as t13, mytable as t14, mytable as t15 WHERE t1.i = t2.i and t2.i = t3.i and t3.i = t4.i and t4.i = t5.i and t5.i = t6.i and t6.i = t7.i and t7.i = t8.i and t8.i = t9.i and t9.i = t10.i and t10.i = t11.i and t11.i = t12.i and t12.i = t13.i and t13.i = t14.i and t14.i = t15.i`, ExpectedErr: sql.ErrUnsupportedJoinFactorCount, }, { Query: `SELECT t1.* FROM mytable as t1 LEFT JOIN mytable as t2 ON t1.i = t2.i LEFT JOIN mytable as t3 ON t2.i = t3.i LEFT JOIN mytable as t4 ON t3.i = t4.i LEFT JOIN mytable as t5 ON t4.i = t5.i LEFT JOIN mytable as t6 ON t5.i = t6.i LEFT JOIN mytable as t7 ON t6.i = t7.i LEFT JOIN mytable as t8 ON t7.i = t8.i LEFT JOIN mytable as t9 ON t8.i = t9.i LEFT JOIN mytable as t10 ON t9.i = t10.i LEFT JOIN mytable as t11 ON t10.i = t11.i LEFT JOIN mytable as t12 ON t11.i = t12.i LEFT JOIN mytable as t13 ON t12.i = t13.i LEFT JOIN mytable as t14 ON t13.i = t14.i LEFT JOIN mytable as t15 ON t14.i = t15.i`, ExpectedErr: sql.ErrUnsupportedJoinFactorCount, }, { Query: `CREATE PROCEDURE proc1 (OUT out_count INT) READS SQL DATA SELECT COUNT(*) FROM mytable WHERE i = 1 AND s = 'first row' AND func1(i);`, ExpectedErr: sql.ErrFunctionNotFound, }, }
var ExplodeQueries = []QueryTest{ { Query: `SELECT a, EXPLODE(b), c FROM explode`, Expected: []sql.Row{ {int64(1), "a", "first"}, {int64(1), "b", "first"}, {int64(2), "c", "second"}, {int64(2), "d", "second"}, {int64(3), "e", "third"}, {int64(3), "f", "third"}, }, }, { Query: `SELECT a, EXPLODE(b) AS x, c FROM explode`, Expected: []sql.Row{ {int64(1), "a", "first"}, {int64(1), "b", "first"}, {int64(2), "c", "second"}, {int64(2), "d", "second"}, {int64(3), "e", "third"}, {int64(3), "f", "third"}, }, }, { Query: `SELECT EXPLODE(SPLIT(c, "")) FROM explode LIMIT 5`, Expected: []sql.Row{ {"f"}, {"i"}, {"r"}, {"s"}, {"t"}, }, }, { Query: `SELECT a, EXPLODE(b) AS x, c FROM explode WHERE x = 'e'`, Expected: []sql.Row{ {int64(3), "e", "third"}, }, }, { Query: `SELECT HEX(UNHEX(375));`, Expected: []sql.Row{ {"0375"}, }, }, }
var ExternalProcedureTests = []ScriptTest{ { Name: "INOUT on first param, IN on second param", SetUpScript: []string{ "SET @outparam = 5;", "CALL memory_inout_add(@outparam, 11);", }, Assertions: []ScriptTestAssertion{ { Query: "SELECT @outparam;", Expected: []sql.Row{{16}}, }, }, }, { Name: "Called from standard stored procedure", SetUpScript: []string{ "CREATE PROCEDURE p1(x BIGINT) BEGIN CALL memory_inout_add(x, x); SELECT x; END;", }, Assertions: []ScriptTestAssertion{ { Query: "CALL p1(11);", Expected: []sql.Row{{22}}, }, }, }, { Name: "Overloaded Name", Assertions: []ScriptTestAssertion{ { Query: "CALL memory_overloaded_mult(1);", Expected: []sql.Row{{1}}, }, { Query: "CALL memory_overloaded_mult(2, 3);", Expected: []sql.Row{{6}}, }, { Query: "CALL memory_overloaded_mult(4, 5, 6);", Expected: []sql.Row{{120}}, }, }, }, { Name: "Passing in all supported types", Assertions: []ScriptTestAssertion{ { Query: "CALL memory_overloaded_type_test(1, 100, 10000, 1000000, 100000000, 3, 300," + "10, 1000, 100000, 10000000, 1000000000, 30, 3000);", Expected: []sql.Row{{1111114444}}, }, { Query: "CALL memory_overloaded_type_test(false, 'hi', 'A', '2020-02-20 12:00:00', 123.456," + "true, 'bye', 'B', '2022-02-02 12:00:00', 654.32);", Expected: []sql.Row{{`aa:false,ba:true,ab:"hi",bb:"bye",ac:[65],bc:[66],ad:2020-02-20,bd:2022-02-02,ae:123.456,be:654.32`}}, }, }, }, { Name: "BOOL and []BYTE INOUT conversions", SetUpScript: []string{ "SET @outparam1 = 1;", "SET @outparam2 = 0;", "SET @outparam3 = 'A';", "SET @outparam4 = 'B';", }, Assertions: []ScriptTestAssertion{ { Query: "SELECT @outparam1, @outparam2, @outparam3, @outparam4;", Expected: []sql.Row{{1, 0, "A", "B"}}, }, { Query: "CALL memory_inout_bool_byte(@outparam1, @outparam2, @outparam3, @outparam4);", Expected: []sql.Row{}, }, { Query: "SELECT @outparam1, @outparam2, @outparam3, @outparam4;", Expected: []sql.Row{{1, 1, "A", "C"}}, }, { Query: "CALL memory_inout_bool_byte(@outparam1, @outparam2, @outparam3, @outparam4);", Expected: []sql.Row{}, }, { Query: "SELECT @outparam1, @outparam2, @outparam3, @outparam4;", Expected: []sql.Row{{1, 0, "A", "D"}}, }, }, }, { Name: "Errors returned", Assertions: []ScriptTestAssertion{ { Query: "CALL memory_error_table_not_found();", ExpectedErr: sql.ErrTableNotFound, }, }, }, { Name: "Variadic parameter", Assertions: []ScriptTestAssertion{ { Query: "CALL memory_variadic_add();", Expected: []sql.Row{{0}}, }, { Query: "CALL memory_variadic_add(1);", Expected: []sql.Row{{1}}, }, { Query: "CALL memory_variadic_add(1, 2);", Expected: []sql.Row{{3}}, }, { Query: "CALL memory_variadic_add(1, 2, 3);", Expected: []sql.Row{{6}}, }, { Query: "CALL memory_variadic_add(1, 2, 3, 4);", Expected: []sql.Row{{10}}, }, }, }, { Name: "Variadic byte slices", Assertions: []ScriptTestAssertion{ { Query: "CALL memory_variadic_byte_slice();", Expected: []sql.Row{{""}}, }, { Query: "CALL memory_variadic_byte_slice('A');", Expected: []sql.Row{{"A"}}, }, { Query: "CALL memory_variadic_byte_slice('A', 'B');", Expected: []sql.Row{{"AB"}}, }, }, }, { Name: "Variadic overloading", Assertions: []ScriptTestAssertion{ { Query: "CALL memory_variadic_overload();", ExpectedErr: sql.ErrCallIncorrectParameterCount, }, { Query: "CALL memory_variadic_overload('A');", ExpectedErr: sql.ErrCallIncorrectParameterCount, }, { Query: "CALL memory_variadic_overload('A', 'B');", Expected: []sql.Row{{"A-B"}}, }, { Query: "CALL memory_variadic_overload('A', 'B', 'C');", ExpectedErr: sql.ErrInvalidValue, }, { Query: "CALL memory_variadic_overload('A', 'B', 5);", Expected: []sql.Row{{"A,B,[5]"}}, }, }, }, }
var ForeignKeyTests = []ScriptTest{ { Name: "ALTER TABLE Single Named FOREIGN KEY", SetUpScript: []string{ "ALTER TABLE child ADD CONSTRAINT fk_named FOREIGN KEY (v1) REFERENCES parent(v1);", }, Assertions: []ScriptTestAssertion{ { Query: "SHOW CREATE TABLE child;", Expected: []sql.Row{{"child", "CREATE TABLE `child` (\n `id` int NOT NULL,\n `v1` int,\n `v2` int,\n PRIMARY KEY (`id`),\n KEY `v1` (`v1`),\n CONSTRAINT `fk_named` FOREIGN KEY (`v1`) REFERENCES `parent` (`v1`)\n) ENGINE=InnoDB DEFAULT CHARSET=utf8mb4 COLLATE=utf8mb4_0900_bin"}}, }, }, }, { Name: "CREATE TABLE Single Named FOREIGN KEY", SetUpScript: []string{ "CREATE TABLE sibling (id int PRIMARY KEY, v1 int, CONSTRAINT fk_named FOREIGN KEY (v1) REFERENCES parent(v1));", }, Assertions: []ScriptTestAssertion{ { Query: "SHOW CREATE TABLE sibling;", Expected: []sql.Row{{"sibling", "CREATE TABLE `sibling` (\n `id` int NOT NULL,\n `v1` int,\n PRIMARY KEY (`id`),\n KEY `v1` (`v1`),\n CONSTRAINT `fk_named` FOREIGN KEY (`v1`) REFERENCES `parent` (`v1`)\n) ENGINE=InnoDB DEFAULT CHARSET=utf8mb4 COLLATE=utf8mb4_0900_bin"}}, }, }, }, { Name: "Parent table index required", Assertions: []ScriptTestAssertion{ { Query: "ALTER TABLE child ADD CONSTRAINT fk1 FOREIGN KEY (v1,v2) REFERENCES parent(v1,v2);", ExpectedErr: sql.ErrForeignKeyMissingReferenceIndex, }, { Query: "ALTER TABLE child ADD CONSTRAINT fk_id FOREIGN KEY (v1) REFERENCES parent(id);", Expected: []sql.Row{{sql.NewOkResult(0)}}, }, }, }, { Name: "CREATE TABLE Name Collision", Assertions: []ScriptTestAssertion{ { Query: "CREATE TABLE child2 (id INT PRIMARY KEY, v1 INT, CONSTRAINT fk_name FOREIGN KEY (v1) REFERENCES parent(v1), CONSTRAINT fk_name FOREIGN KEY (v1) REFERENCES parent(v1));", ExpectedErr: sql.ErrForeignKeyDuplicateName, }, }, }, { Name: "CREATE TABLE Type Mismatch", SetUpScript: []string{ "CREATE TABLE sibling (pk INT PRIMARY KEY, v1 TIME);", }, Assertions: []ScriptTestAssertion{ { Query: "ALTER TABLE sibling ADD CONSTRAINT fk1 FOREIGN KEY (v1) REFERENCES parent(v1);", ExpectedErr: sql.ErrForeignKeyColumnTypeMismatch, }, }, }, { Name: "CREATE TABLE Type Mismatch special case for strings", SetUpScript: []string{ "CREATE TABLE parent1 (pk BIGINT PRIMARY KEY, v1 CHAR(20), INDEX (v1));", "CREATE TABLE parent2 (pk BIGINT PRIMARY KEY, v1 VARCHAR(20), INDEX (v1));", "CREATE TABLE parent3 (pk BIGINT PRIMARY KEY, v1 BINARY(20), INDEX (v1));", "CREATE TABLE parent4 (pk BIGINT PRIMARY KEY, v1 VARBINARY(20), INDEX (v1));", }, Assertions: []ScriptTestAssertion{ { Query: "CREATE TABLE child1 (pk BIGINT PRIMARY KEY, v1 CHAR(30), CONSTRAINT fk_child1 FOREIGN KEY (v1) REFERENCES parent1 (v1));", Expected: []sql.Row{{sql.NewOkResult(0)}}, }, { Query: "CREATE TABLE child2 (pk BIGINT PRIMARY KEY, v1 VARCHAR(30), CONSTRAINT fk_child2 FOREIGN KEY (v1) REFERENCES parent2 (v1));", Expected: []sql.Row{{sql.NewOkResult(0)}}, }, { Query: "CREATE TABLE child3 (pk BIGINT PRIMARY KEY, v1 BINARY(30), CONSTRAINT fk_child3 FOREIGN KEY (v1) REFERENCES parent3 (v1));", Expected: []sql.Row{{sql.NewOkResult(0)}}, }, { Query: "CREATE TABLE child4 (pk BIGINT PRIMARY KEY, v1 VARBINARY(30), CONSTRAINT fk_child4 FOREIGN KEY (v1) REFERENCES parent4 (v1));", Expected: []sql.Row{{sql.NewOkResult(0)}}, }, }, }, { Name: "CREATE TABLE Key Count Mismatch", Assertions: []ScriptTestAssertion{ { Query: "ALTER TABLE child ADD CONSTRAINT fk1 FOREIGN KEY (v1) REFERENCES parent(v1, v2);", ExpectedErr: sql.ErrForeignKeyColumnCountMismatch, }, { Query: "ALTER TABLE child ADD CONSTRAINT fk1 FOREIGN KEY (v1, v2) REFERENCES parent(v1);", ExpectedErr: sql.ErrForeignKeyColumnCountMismatch, }, }, }, { Name: "SET DEFAULT not supported", Assertions: []ScriptTestAssertion{ { Query: "ALTER TABLE child ADD CONSTRAINT fk1 FOREIGN KEY (v1) REFERENCES parent(v1) ON DELETE SET DEFAULT;", ExpectedErr: sql.ErrForeignKeySetDefault, }, { Query: "ALTER TABLE child ADD CONSTRAINT fk1 FOREIGN KEY (v1) REFERENCES parent(v1) ON UPDATE SET DEFAULT;", ExpectedErr: sql.ErrForeignKeySetDefault, }, { Query: "ALTER TABLE child ADD CONSTRAINT fk1 FOREIGN KEY (v1) REFERENCES parent(v1) ON UPDATE SET DEFAULT ON DELETE SET DEFAULT;", ExpectedErr: sql.ErrForeignKeySetDefault, }, }, }, { Name: "CREATE TABLE Disallow TEXT/BLOB", SetUpScript: []string{ "CREATE TABLE parent1 (id INT PRIMARY KEY, v1 TINYTEXT, v2 TEXT, v3 MEDIUMTEXT, v4 LONGTEXT);", "CREATE TABLE parent2 (id INT PRIMARY KEY, v1 TINYBLOB, v2 BLOB, v3 MEDIUMBLOB, v4 LONGBLOB);", }, Assertions: []ScriptTestAssertion{ { Query: "CREATE TABLE child11 (id INT PRIMARY KEY, parent_v1 TINYTEXT, FOREIGN KEY (parent_v1) REFERENCES parent1(v1));", ExpectedErr: sql.ErrForeignKeyTextBlob, }, { Query: "CREATE TABLE child12 (id INT PRIMARY KEY, parent_v2 TEXT, FOREIGN KEY (parent_v2) REFERENCES parent1(v2));", ExpectedErr: sql.ErrForeignKeyTextBlob, }, { Query: "CREATE TABLE child13 (id INT PRIMARY KEY, parent_v3 MEDIUMTEXT, FOREIGN KEY (parent_v3) REFERENCES parent1(v3));", ExpectedErr: sql.ErrForeignKeyTextBlob, }, { Query: "CREATE TABLE child14 (id INT PRIMARY KEY, parent_v4 LONGTEXT, FOREIGN KEY (parent_v4) REFERENCES parent1(v4));", ExpectedErr: sql.ErrForeignKeyTextBlob, }, { Query: "CREATE TABLE child21 (id INT PRIMARY KEY, parent_v1 TINYBLOB, FOREIGN KEY (parent_v1) REFERENCES parent2(v1));", ExpectedErr: sql.ErrForeignKeyTextBlob, }, { Query: "CREATE TABLE child22 (id INT PRIMARY KEY, parent_v2 BLOB, FOREIGN KEY (parent_v2) REFERENCES parent2(v2));", ExpectedErr: sql.ErrForeignKeyTextBlob, }, { Query: "CREATE TABLE child23 (id INT PRIMARY KEY, parent_v3 MEDIUMBLOB, FOREIGN KEY (parent_v3) REFERENCES parent2(v3));", ExpectedErr: sql.ErrForeignKeyTextBlob, }, { Query: "CREATE TABLE child24 (id INT PRIMARY KEY, parent_v4 LONGBLOB, FOREIGN KEY (parent_v4) REFERENCES parent2(v4));", ExpectedErr: sql.ErrForeignKeyTextBlob, }, }, }, { Name: "CREATE TABLE Non-existent Table", Assertions: []ScriptTestAssertion{ { Query: "ALTER TABLE child ADD CONSTRAINT fk1 FOREIGN KEY (v1) REFERENCES father(v1);", ExpectedErr: sql.ErrTableNotFound, }, }, }, { Name: "CREATE TABLE Non-existent Columns", Assertions: []ScriptTestAssertion{ { Query: "ALTER TABLE child ADD CONSTRAINT fk1 FOREIGN KEY (random) REFERENCES parent(v1);", ExpectedErr: sql.ErrTableColumnNotFound, }, { Query: "ALTER TABLE child ADD CONSTRAINT fk1 FOREIGN KEY (v1) REFERENCES parent(random);", ExpectedErr: sql.ErrTableColumnNotFound, }, }, }, { Name: "ALTER TABLE Foreign Key Name Collision", SetUpScript: []string{ "ALTER TABLE child ADD CONSTRAINT fk_name FOREIGN KEY (v1) REFERENCES parent(v1);", }, Assertions: []ScriptTestAssertion{ { Query: "ALTER TABLE child ADD CONSTRAINT fk_name FOREIGN KEY (v1) REFERENCES parent(v1);", ExpectedErr: sql.ErrForeignKeyDuplicateName, }, }, }, { Name: "ALTER TABLE DROP FOREIGN KEY", SetUpScript: []string{ "ALTER TABLE child ADD CONSTRAINT fk_name FOREIGN KEY (v1) REFERENCES parent(v1);", }, Assertions: []ScriptTestAssertion{ { Query: "SHOW CREATE TABLE child;", Expected: []sql.Row{{"child", "CREATE TABLE `child` (\n `id` int NOT NULL,\n `v1` int,\n `v2` int,\n PRIMARY KEY (`id`),\n KEY `v1` (`v1`),\n CONSTRAINT `fk_name` FOREIGN KEY (`v1`) REFERENCES `parent` (`v1`)\n) ENGINE=InnoDB DEFAULT CHARSET=utf8mb4 COLLATE=utf8mb4_0900_bin"}}, }, { Query: "ALTER TABLE child DROP FOREIGN KEY fk_name;", Expected: []sql.Row{{sql.NewOkResult(0)}}, }, { Query: "SHOW CREATE TABLE child;", Expected: []sql.Row{{"child", "CREATE TABLE `child` (\n `id` int NOT NULL,\n `v1` int,\n `v2` int,\n PRIMARY KEY (`id`),\n KEY `v1` (`v1`)\n) ENGINE=InnoDB DEFAULT CHARSET=utf8mb4 COLLATE=utf8mb4_0900_bin"}}, }, { Query: "ALTER TABLE child DROP FOREIGN KEY fk_name;", ExpectedErr: sql.ErrForeignKeyNotFound, }, }, }, { Name: "ALTER TABLE SET NULL on non-nullable column", SetUpScript: []string{ "ALTER TABLE child MODIFY v1 int NOT NULL;", }, Assertions: []ScriptTestAssertion{ { Query: "ALTER TABLE child ADD CONSTRAINT fk1 FOREIGN KEY (v1) REFERENCES parent(v1) ON DELETE SET NULL;", ExpectedErr: sql.ErrForeignKeySetNullNonNullable, }, { Query: "ALTER TABLE child ADD CONSTRAINT fk1 FOREIGN KEY (v1) REFERENCES parent(v1) ON UPDATE SET NULL;", ExpectedErr: sql.ErrForeignKeySetNullNonNullable, }, { Query: "ALTER TABLE child ADD CONSTRAINT fk1 FOREIGN KEY (v1) REFERENCES parent(v1) ON DELETE SET NULL ON UPDATE SET NULL;", ExpectedErr: sql.ErrForeignKeySetNullNonNullable, }, }, }, { Name: "ADD FOREIGN KEY fails on existing table when data would cause violation", SetUpScript: []string{ "INSERT INTO parent VALUES (1, 1, 1), (2, 2, 2);", "INSERT INTO child VALUES (1, 1, 1), (2, 3, 2);", }, Assertions: []ScriptTestAssertion{ { Query: "ALTER TABLE child ADD CONSTRAINT fk_name FOREIGN KEY (v1) REFERENCES parent(v1)", ExpectedErr: sql.ErrForeignKeyChildViolation, }, }, }, { Name: "RENAME TABLE", SetUpScript: []string{ "ALTER TABLE child ADD CONSTRAINT fk_name FOREIGN KEY (v1) REFERENCES parent(v1);", "RENAME TABLE parent TO new_parent;", }, Assertions: []ScriptTestAssertion{ { Query: "SHOW CREATE TABLE child;", Expected: []sql.Row{{"child", "CREATE TABLE `child` (\n `id` int NOT NULL,\n `v1` int,\n `v2` int,\n PRIMARY KEY (`id`),\n KEY `v1` (`v1`),\n CONSTRAINT `fk_name` FOREIGN KEY (`v1`) REFERENCES `new_parent` (`v1`)\n) ENGINE=InnoDB DEFAULT CHARSET=utf8mb4 COLLATE=utf8mb4_0900_bin"}}, }, { Query: "RENAME TABLE child TO new_child;", Expected: []sql.Row{{sql.NewOkResult(0)}}, }, { Query: "SHOW CREATE TABLE new_child;", Expected: []sql.Row{{"new_child", "CREATE TABLE `new_child` (\n `id` int NOT NULL,\n `v1` int,\n `v2` int,\n PRIMARY KEY (`id`),\n KEY `v1` (`v1`),\n CONSTRAINT `fk_name` FOREIGN KEY (`v1`) REFERENCES `new_parent` (`v1`)\n) ENGINE=InnoDB DEFAULT CHARSET=utf8mb4 COLLATE=utf8mb4_0900_bin"}}, }, }, }, { Name: "DROP TABLE", SetUpScript: []string{ "ALTER TABLE child ADD CONSTRAINT fk_name FOREIGN KEY (v1) REFERENCES parent(v1);", }, Assertions: []ScriptTestAssertion{ { Query: "DROP TABLE parent;", ExpectedErr: sql.ErrForeignKeyDropTable, }, { Query: "DROP TABLE child;", Expected: []sql.Row{{sql.NewOkResult(0)}}, }, { Query: "DROP TABLE parent;", Expected: []sql.Row{{sql.NewOkResult(0)}}, }, }, }, { Name: "Indexes used by foreign keys can't be dropped", SetUpScript: []string{ "ALTER TABLE child ADD INDEX v1 (v1);", "ALTER TABLE child ADD CONSTRAINT fk_name FOREIGN KEY (v1) REFERENCES parent(v1);", }, Assertions: []ScriptTestAssertion{ { Query: "ALTER TABLE child DROP INDEX v1;", ExpectedErr: sql.ErrForeignKeyDropIndex, }, { Query: "ALTER TABLE parent DROP INDEX v1;", ExpectedErr: sql.ErrForeignKeyDropIndex, }, { Query: "ALTER TABLE child DROP FOREIGN KEY fk_name;", Expected: []sql.Row{{sql.NewOkResult(0)}}, }, { Query: "ALTER TABLE child DROP INDEX v1;", Expected: []sql.Row{{sql.NewOkResult(0)}}, }, { Query: "ALTER TABLE parent DROP INDEX v1;", Expected: []sql.Row{{sql.NewOkResult(0)}}, }, }, }, { Name: "ALTER TABLE RENAME COLUMN", SetUpScript: []string{ "ALTER TABLE child ADD CONSTRAINT fk1 FOREIGN KEY (v1) REFERENCES parent(v1);", "ALTER TABLE parent RENAME COLUMN v1 TO v1_new;", "ALTER TABLE child RENAME COLUMN v1 TO v1_new;", }, Assertions: []ScriptTestAssertion{ { Query: "SHOW CREATE TABLE child;", Expected: []sql.Row{{"child", "CREATE TABLE `child` (\n `id` int NOT NULL,\n `v1_new` int,\n `v2` int,\n PRIMARY KEY (`id`),\n KEY `v1` (`v1_new`),\n CONSTRAINT `fk1` FOREIGN KEY (`v1_new`) REFERENCES `parent` (`v1_new`)\n) ENGINE=InnoDB DEFAULT CHARSET=utf8mb4 COLLATE=utf8mb4_0900_bin"}}, }, }, }, { Name: "ALTER TABLE MODIFY COLUMN type change not allowed", SetUpScript: []string{ "ALTER TABLE child ADD CONSTRAINT fk1 FOREIGN KEY (v1) REFERENCES parent(v1);", }, Assertions: []ScriptTestAssertion{ { Query: "ALTER TABLE parent MODIFY v1 MEDIUMINT;", ExpectedErr: sql.ErrForeignKeyTypeChange, }, { Query: "ALTER TABLE child MODIFY v1 MEDIUMINT;", ExpectedErr: sql.ErrForeignKeyTypeChange, }, }, }, { Name: "ALTER TABLE MODIFY COLUMN type change allowed when lengthening string", SetUpScript: []string{ "CREATE TABLE parent1 (pk BIGINT PRIMARY KEY, v1 CHAR(20), INDEX (v1));", "CREATE TABLE parent2 (pk BIGINT PRIMARY KEY, v1 VARCHAR(20), INDEX (v1));", "CREATE TABLE parent3 (pk BIGINT PRIMARY KEY, v1 BINARY(20), INDEX (v1));", "CREATE TABLE parent4 (pk BIGINT PRIMARY KEY, v1 VARBINARY(20), INDEX (v1));", "CREATE TABLE child1 (pk BIGINT PRIMARY KEY, v1 CHAR(20), CONSTRAINT fk_child1 FOREIGN KEY (v1) REFERENCES parent1 (v1));", "CREATE TABLE child2 (pk BIGINT PRIMARY KEY, v1 VARCHAR(20), CONSTRAINT fk_child2 FOREIGN KEY (v1) REFERENCES parent2 (v1));", "CREATE TABLE child3 (pk BIGINT PRIMARY KEY, v1 BINARY(20), CONSTRAINT fk_child3 FOREIGN KEY (v1) REFERENCES parent3 (v1));", "CREATE TABLE child4 (pk BIGINT PRIMARY KEY, v1 VARBINARY(20), CONSTRAINT fk_child4 FOREIGN KEY (v1) REFERENCES parent4 (v1));", "INSERT INTO parent2 VALUES (1, 'aa'), (2, 'bb');", "INSERT INTO child2 VALUES (1, 'aa');", }, Assertions: []ScriptTestAssertion{ { Query: "ALTER TABLE parent1 MODIFY v1 CHAR(10);", ExpectedErr: sql.ErrForeignKeyTypeChange, }, { Query: "ALTER TABLE child1 MODIFY v1 CHAR(10);", ExpectedErr: sql.ErrForeignKeyTypeChange, }, { Query: "ALTER TABLE parent2 MODIFY v1 VARCHAR(10);", ExpectedErr: sql.ErrForeignKeyTypeChange, }, { Query: "ALTER TABLE child2 MODIFY v1 VARCHAR(10);", ExpectedErr: sql.ErrForeignKeyTypeChange, }, { Query: "ALTER TABLE parent3 MODIFY v1 BINARY(10);", ExpectedErr: sql.ErrForeignKeyTypeChange, }, { Query: "ALTER TABLE child3 MODIFY v1 BINARY(10);", ExpectedErr: sql.ErrForeignKeyTypeChange, }, { Query: "ALTER TABLE parent4 MODIFY v1 VARBINARY(10);", ExpectedErr: sql.ErrForeignKeyTypeChange, }, { Query: "ALTER TABLE child4 MODIFY v1 VARBINARY(10);", ExpectedErr: sql.ErrForeignKeyTypeChange, }, { Query: "ALTER TABLE parent1 MODIFY v1 CHAR(30);", Expected: []sql.Row{{sql.NewOkResult(0)}}, }, { Query: "ALTER TABLE child1 MODIFY v1 CHAR(30);", Expected: []sql.Row{{sql.NewOkResult(0)}}, }, { Query: "ALTER TABLE parent2 MODIFY v1 VARCHAR(30);", Expected: []sql.Row{{sql.NewOkResult(0)}}, }, { Query: "ALTER TABLE child2 MODIFY v1 VARCHAR(30);", Expected: []sql.Row{{sql.NewOkResult(0)}}, }, { Query: "ALTER TABLE parent3 MODIFY v1 BINARY(30);", Expected: []sql.Row{{sql.NewOkResult(0)}}, }, { Query: "ALTER TABLE child3 MODIFY v1 BINARY(30);", Expected: []sql.Row{{sql.NewOkResult(0)}}, }, { Query: "ALTER TABLE parent4 MODIFY v1 VARBINARY(30);", Expected: []sql.Row{{sql.NewOkResult(0)}}, }, { Query: "ALTER TABLE child4 MODIFY v1 VARBINARY(30);", Expected: []sql.Row{{sql.NewOkResult(0)}}, }, { Query: "INSERT INTO child2 VALUES (2, 'bb');", Expected: []sql.Row{{sql.NewOkResult(1)}}, }, { Query: "INSERT INTO child2 VALUES (3, 'cc');", ExpectedErr: sql.ErrForeignKeyChildViolation, }, }, }, { Name: "ALTER TABLE MODIFY COLUMN type change only cares about foreign key columns", SetUpScript: []string{ "CREATE TABLE parent1 (pk INT PRIMARY KEY, v1 INT UNSIGNED, v2 INT UNSIGNED, INDEX (v1));", "CREATE TABLE child1 (pk INT PRIMARY KEY, v1 INT UNSIGNED, v2 INT UNSIGNED, CONSTRAINT fk_name FOREIGN KEY (v1) REFERENCES parent1(v1));", "INSERT INTO parent1 VALUES (1, 2, 3), (4, 5, 6);", "INSERT INTO child1 VALUES (7, 2, 9);", }, Assertions: []ScriptTestAssertion{ { Query: "ALTER TABLE parent1 MODIFY v1 BIGINT;", ExpectedErr: sql.ErrForeignKeyTypeChange, }, { Query: "ALTER TABLE child1 MODIFY v1 BIGINT;", ExpectedErr: sql.ErrForeignKeyTypeChange, }, { Query: "ALTER TABLE parent1 MODIFY v2 BIGINT;", Expected: []sql.Row{{sql.NewOkResult(0)}}, }, { Query: "ALTER TABLE child1 MODIFY v2 BIGINT;", Expected: []sql.Row{{sql.NewOkResult(0)}}, }, }, }, { Name: "DROP COLUMN parent", SetUpScript: []string{ "ALTER TABLE child ADD CONSTRAINT fk_name FOREIGN KEY (v1) REFERENCES parent(v1);", }, Assertions: []ScriptTestAssertion{ { Query: "ALTER TABLE parent DROP COLUMN v1;", ExpectedErr: sql.ErrForeignKeyDropColumn, }, { Query: "ALTER TABLE child DROP FOREIGN KEY fk_name;", Expected: []sql.Row{{sql.NewOkResult(0)}}, }, { Query: "ALTER TABLE parent DROP COLUMN v1;", Expected: []sql.Row{{sql.NewOkResult(0)}}, }, }, }, { Name: "DROP COLUMN child", SetUpScript: []string{ "ALTER TABLE child ADD CONSTRAINT fk_name FOREIGN KEY (v1) REFERENCES parent(v1);", }, Assertions: []ScriptTestAssertion{ { Query: "ALTER TABLE child DROP COLUMN v1;", ExpectedErr: sql.ErrForeignKeyDropColumn, }, { Query: "ALTER TABLE child DROP FOREIGN KEY fk_name;", Expected: []sql.Row{{sql.NewOkResult(0)}}, }, { Query: "ALTER TABLE child DROP COLUMN v1;", Expected: []sql.Row{{sql.NewOkResult(0)}}, }, }, }, { Name: "Disallow change column to nullable with ON UPDATE SET NULL", SetUpScript: []string{ "ALTER TABLE child ADD CONSTRAINT fk_name FOREIGN KEY (v1) REFERENCES parent(v1) ON UPDATE SET NULL", }, Assertions: []ScriptTestAssertion{ { Query: "ALTER TABLE child CHANGE COLUMN v1 v1 INT NOT NULL;", ExpectedErr: sql.ErrForeignKeyTypeChangeSetNull, }, }, }, { Name: "Disallow change column to nullable with ON DELETE SET NULL", SetUpScript: []string{ "ALTER TABLE child ADD CONSTRAINT fk_name FOREIGN KEY (v1) REFERENCES parent(v1) ON DELETE SET NULL", }, Assertions: []ScriptTestAssertion{ { Query: "ALTER TABLE child CHANGE COLUMN v1 v1 INT NOT NULL;", ExpectedErr: sql.ErrForeignKeyTypeChangeSetNull, }, }, }, { Name: "SQL CASCADE", SetUpScript: []string{ "CREATE TABLE one (pk BIGINT PRIMARY KEY, v1 BIGINT, v2 BIGINT, INDEX v1 (v1));", "CREATE TABLE two (pk BIGINT PRIMARY KEY, v1 BIGINT, v2 BIGINT, INDEX v1v2 (v1, v2), CONSTRAINT fk_name_1 FOREIGN KEY (v1) REFERENCES one(v1) ON DELETE CASCADE ON UPDATE CASCADE);", "CREATE TABLE three (pk BIGINT PRIMARY KEY, v1 BIGINT, v2 BIGINT, CONSTRAINT fk_name_2 FOREIGN KEY (v1, v2) REFERENCES two(v1, v2) ON DELETE CASCADE ON UPDATE CASCADE);", "INSERT INTO one VALUES (1, 1, 4), (2, 2, 5), (3, 3, 6), (4, 4, 5);", "INSERT INTO two VALUES (2, 1, 1), (3, 2, 2), (4, 3, 3), (5, 4, 4);", "INSERT INTO three VALUES (3, 1, 1), (4, 2, 2), (5, 3, 3), (6, 4, 4);", "UPDATE one SET v1 = v1 + v2;", "DELETE FROM one WHERE pk = 3;", "UPDATE two SET v2 = v1 - 2;", }, Assertions: []ScriptTestAssertion{ { Query: "SELECT * FROM one;", Expected: []sql.Row{{1, 5, 4}, {2, 7, 5}, {4, 9, 5}}, }, { Query: "SELECT * FROM two;", Expected: []sql.Row{{2, 5, 3}, {3, 7, 5}}, }, { Query: "SELECT * FROM three;", Expected: []sql.Row{{3, 5, 3}, {4, 7, 5}}, }, }, }, { Name: "SQL SET NULL", SetUpScript: []string{ "CREATE TABLE one (pk BIGINT PRIMARY KEY, v1 BIGINT, v2 BIGINT, INDEX v1 (v1));", "CREATE TABLE two (pk BIGINT PRIMARY KEY, v1 BIGINT, v2 BIGINT, CONSTRAINT fk_name_1 FOREIGN KEY (v1) REFERENCES one(v1) ON DELETE SET NULL ON UPDATE SET NULL);", "INSERT INTO one VALUES (1, 1, 1), (2, 2, 2), (3, 3, 3);", "INSERT INTO two VALUES (1, 1, 1), (2, 2, 2), (3, 3, 3);", "UPDATE one SET v1 = v1 * v2;", "INSERT INTO one VALUES (4, 4, 4);", "INSERT INTO two VALUES (4, 4, 4);", "UPDATE one SET v2 = v1 * v2;", }, Assertions: []ScriptTestAssertion{ { Query: "SELECT * FROM one;", Expected: []sql.Row{{1, 1, 1}, {2, 4, 8}, {3, 9, 27}, {4, 4, 16}}, }, { Query: "SELECT * FROM two;", Expected: []sql.Row{{1, 1, 1}, {2, nil, 2}, {3, nil, 3}, {4, 4, 4}}, }, }, }, { Name: "SQL RESTRICT", SetUpScript: []string{ "CREATE TABLE one (pk BIGINT PRIMARY KEY, v1 BIGINT, v2 BIGINT, INDEX v1 (v1));", "CREATE TABLE two (pk BIGINT PRIMARY KEY, v1 BIGINT, v2 BIGINT, CONSTRAINT fk_name_1 FOREIGN KEY (v1) REFERENCES one(v1) ON DELETE RESTRICT ON UPDATE RESTRICT);", "INSERT INTO one VALUES (1, 1, 1), (2, 2, 2), (3, 3, 3);", "INSERT INTO two VALUES (1, 1, 1), (2, 2, 2), (3, 3, 3);", }, Assertions: []ScriptTestAssertion{ { Query: "UPDATE one SET v1 = v1 + v2;", ExpectedErr: sql.ErrForeignKeyParentViolation, }, { Query: "UPDATE one SET v1 = v1;", Expected: []sql.Row{{sql.OkResult{Info: plan.UpdateInfo{Matched: 3}}}}, }, { Query: "DELETE FROM one;", ExpectedErr: sql.ErrForeignKeyParentViolation, }, }, }, { Name: "SQL no reference options", SetUpScript: []string{ "CREATE TABLE one (pk BIGINT PRIMARY KEY, v1 BIGINT, v2 BIGINT, INDEX v1 (v1));", "CREATE TABLE two (pk BIGINT PRIMARY KEY, v1 BIGINT, v2 BIGINT, CONSTRAINT fk_name_1 FOREIGN KEY (v1) REFERENCES one(v1));", "INSERT INTO one VALUES (1, 1, 1), (2, 2, 2), (3, 3, 3);", "INSERT INTO two VALUES (1, 1, 1), (2, 2, 2), (3, 3, 3);", }, Assertions: []ScriptTestAssertion{ { Query: "UPDATE one SET v1 = v1 + v2;", ExpectedErr: sql.ErrForeignKeyParentViolation, }, { Query: "UPDATE one SET v1 = v1;", Expected: []sql.Row{{sql.OkResult{Info: plan.UpdateInfo{Matched: 3}}}}, }, { Query: "DELETE FROM one;", ExpectedErr: sql.ErrForeignKeyParentViolation, }, }, }, { Name: "SQL INSERT multiple keys violates only one", SetUpScript: []string{ "CREATE TABLE one (pk BIGINT PRIMARY KEY, v1 BIGINT, v2 BIGINT, INDEX v1 (v1), INDEX v2 (v2));", "CREATE TABLE two (pk BIGINT PRIMARY KEY, v1 BIGINT, v2 BIGINT, CONSTRAINT fk_name_1 FOREIGN KEY (v1) REFERENCES one(v1), CONSTRAINT fk_name_2 FOREIGN KEY (v2) REFERENCES one(v2));", "INSERT INTO one VALUES (1, 1, 1), (2, 2, 2), (3, 3, 3);", "INSERT INTO two VALUES (1, NULL, 1);", }, Assertions: []ScriptTestAssertion{ { Query: "INSERT INTO two VALUES (2, NULL, 4);", ExpectedErr: sql.ErrForeignKeyChildViolation, }, { Query: "INSERT INTO two VALUES (3, 4, NULL);", ExpectedErr: sql.ErrForeignKeyChildViolation, }, { Query: "INSERT INTO two VALUES (4, NULL, NULL);", Expected: []sql.Row{{sql.NewOkResult(1)}}, }, }, }, { Name: "Self-referential same column(s)", SetUpScript: []string{ "CREATE INDEX v1v2 ON parent(v1, v2);", "CREATE TABLE parent2 (id INT PRIMARY KEY, v1 INT, v2 INT, INDEX v1v2 (v1, v2));", }, Assertions: []ScriptTestAssertion{ { Query: "ALTER TABLE parent ADD CONSTRAINT fk_name1 FOREIGN KEY (v1) REFERENCES parent(v1);", Expected: []sql.Row{{sql.NewOkResult(0)}}, }, { Query: "ALTER TABLE parent ADD CONSTRAINT fk_name2 FOREIGN KEY (v1, v2) REFERENCES parent(v1, v2);", Expected: []sql.Row{{sql.NewOkResult(0)}}, }, }, }, { Name: "Self-referential child column follows parent RESTRICT", SetUpScript: []string{ "ALTER TABLE parent ADD CONSTRAINT fk_named FOREIGN KEY (v2) REFERENCES parent(v1);", "INSERT INTO parent VALUES (1, 1, 1), (2, 2, 1), (3, 3, NULL);", "UPDATE parent SET v1 = 1 WHERE id = 1;", "UPDATE parent SET v1 = 4 WHERE id = 3;", "DELETE FROM parent WHERE id = 3;", }, Assertions: []ScriptTestAssertion{ { Query: "SELECT * FROM parent;", Expected: []sql.Row{{1, 1, 1}, {2, 2, 1}}, }, { Query: "DELETE FROM parent WHERE v1 = 1;", ExpectedErr: sql.ErrForeignKeyParentViolation, }, { Query: "UPDATE parent SET v1 = 2;", ExpectedErr: sql.ErrForeignKeyParentViolation, }, { Query: "REPLACE INTO parent VALUES (1, 1, 1);", ExpectedErr: sql.ErrForeignKeyParentViolation, }, }, }, { Name: "Self-referential child column follows parent CASCADE", SetUpScript: []string{ "ALTER TABLE parent ADD CONSTRAINT fk_named FOREIGN KEY (v2) REFERENCES parent(v1) ON UPDATE CASCADE ON DELETE CASCADE;", "INSERT INTO parent VALUES (1, 1, 1), (2, 2, 1), (3, 3, NULL);", "UPDATE parent SET v1 = 1 WHERE id = 1;", "UPDATE parent SET v1 = 4 WHERE id = 3;", "DELETE FROM parent WHERE id = 3;", }, Assertions: []ScriptTestAssertion{ { Query: "UPDATE parent SET v1 = 2;", ExpectedErr: sql.ErrForeignKeyParentViolation, }, { Query: "REPLACE INTO parent VALUES (1, 1, 1), (2, 2, 2);", Expected: []sql.Row{{sql.NewOkResult(3)}}, }, { Query: "SELECT * FROM parent;", Expected: []sql.Row{{1, 1, 1}, {2, 2, 2}}, }, { Query: "UPDATE parent SET v1 = 2;", ExpectedErr: sql.ErrForeignKeyParentViolation, }, { Query: "UPDATE parent SET v1 = 2 WHERE id = 1;", ExpectedErr: sql.ErrForeignKeyParentViolation, }, { Query: "REPLACE INTO parent VALUES (1, 1, 2), (2, 2, 1);", ExpectedErr: sql.ErrForeignKeyChildViolation, }, { Query: "UPDATE parent SET v2 = 2 WHERE id = 1;", Expected: []sql.Row{{sql.OkResult{RowsAffected: 1, Info: plan.UpdateInfo{Matched: 1, Updated: 1}}}}, }, { Query: "UPDATE parent SET v2 = 1 WHERE id = 2;", Expected: []sql.Row{{sql.OkResult{RowsAffected: 1, Info: plan.UpdateInfo{Matched: 1, Updated: 1}}}}, }, { Query: "SELECT * FROM parent;", Expected: []sql.Row{{1, 1, 2}, {2, 2, 1}}, }, { Query: "UPDATE parent SET v1 = 2;", ExpectedErr: sql.ErrForeignKeyParentViolation, }, { Query: "UPDATE parent SET v1 = 2 WHERE id = 1;", ExpectedErr: sql.ErrForeignKeyParentViolation, }, { Query: "DELETE FROM parent WHERE v1 = 1;", Expected: []sql.Row{{sql.NewOkResult(1)}}, }, { Query: "SELECT * FROM parent;", Expected: []sql.Row{}, }, }, }, { Name: "Self-referential child column follows parent SET NULL", SetUpScript: []string{ "ALTER TABLE parent ADD CONSTRAINT fk_named FOREIGN KEY (v2) REFERENCES parent(v1) ON UPDATE SET NULL ON DELETE SET NULL;", "INSERT INTO parent VALUES (1,1,1), (2, 2, 1), (3, 3, NULL);", "UPDATE parent SET v1 = 1 WHERE id = 1;", "UPDATE parent SET v1 = 4 WHERE id = 3;", "DELETE FROM parent WHERE id = 3;", }, Assertions: []ScriptTestAssertion{ { Query: "UPDATE parent SET v1 = 2;", ExpectedErr: sql.ErrForeignKeyParentViolation, }, { Query: "REPLACE INTO parent VALUES (1, 1, 1), (2, 2, 2);", Expected: []sql.Row{{sql.NewOkResult(4)}}, }, { Query: "SELECT * FROM parent;", Expected: []sql.Row{{1, 1, 1}, {2, 2, 2}}, }, { Query: "UPDATE parent SET v1 = 2;", ExpectedErr: sql.ErrForeignKeyParentViolation, }, { Query: "UPDATE parent SET v1 = 2 WHERE id = 1;", ExpectedErr: sql.ErrForeignKeyParentViolation, }, { Query: "REPLACE INTO parent VALUES (1,1,2), (2,2,1);", Expected: []sql.Row{{sql.NewOkResult(4)}}, }, { Query: "SELECT * FROM parent;", Expected: []sql.Row{{1, 1, nil}, {2, 2, 1}}, }, { Query: "UPDATE parent SET v2 = 2 WHERE id = 1;", Expected: []sql.Row{{sql.OkResult{RowsAffected: 1, Info: plan.UpdateInfo{Matched: 1, Updated: 1}}}}, }, { Query: "UPDATE parent SET v2 = 1 WHERE id = 2;", Expected: []sql.Row{{sql.OkResult{RowsAffected: 0, Info: plan.UpdateInfo{Matched: 1}}}}, }, { Query: "SELECT * FROM parent;", Expected: []sql.Row{{1, 1, 2}, {2, 2, 1}}, }, { Query: "UPDATE parent SET v1 = 2;", ExpectedErr: sql.ErrForeignKeyParentViolation, }, { Query: "UPDATE parent SET v1 = 2 WHERE id = 1;", ExpectedErr: sql.ErrForeignKeyParentViolation, }, { Query: "DELETE FROM parent WHERE v1 = 1;", Expected: []sql.Row{{sql.NewOkResult(1)}}, }, { Query: "SELECT * FROM parent;", Expected: []sql.Row{{2, 2, nil}}, }, }, }, { Name: "VARCHAR child violation detection", SetUpScript: []string{ "CREATE TABLE colors (id INT NOT NULL, color VARCHAR(32) NOT NULL, PRIMARY KEY (id), INDEX color_index(color));", "CREATE TABLE objects (id INT NOT NULL, name VARCHAR(64) NOT NULL, color VARCHAR(32), PRIMARY KEY(id), CONSTRAINT color_fk FOREIGN KEY (color) REFERENCES colors(color));", "INSERT INTO colors (id, color) VALUES (1, 'red'), (2, 'green'), (3, 'blue'), (4, 'purple');", "INSERT INTO objects (id, name, color) VALUES (1, 'truck', 'red'), (2, 'ball', 'green'), (3, 'shoe', 'blue');", }, Assertions: []ScriptTestAssertion{ { Query: "DELETE FROM colors where color='green';", ExpectedErr: sql.ErrForeignKeyParentViolation, }, { Query: "SELECT * FROM colors;", Expected: []sql.Row{{1, "red"}, {2, "green"}, {3, "blue"}, {4, "purple"}}, }, }, }, { Name: "INSERT IGNORE INTO works correctly with foreign key violations", SetUpScript: []string{ "CREATE TABLE colors (id INT NOT NULL, color VARCHAR(32) NOT NULL, PRIMARY KEY (id), INDEX color_index(color));", "CREATE TABLE objects (id INT NOT NULL, name VARCHAR(64) NOT NULL, color VARCHAR(32), PRIMARY KEY(id), CONSTRAINT color_fk FOREIGN KEY (color) REFERENCES colors(color));", "INSERT INTO colors (id, color) VALUES (1, 'red'), (2, 'green'), (3, 'blue'), (4, 'purple');", "INSERT INTO objects (id, name, color) VALUES (1, 'truck', 'red'), (2, 'ball', 'green'), (3, 'shoe', 'blue');", }, Assertions: []ScriptTestAssertion{ { Query: "INSERT IGNORE INTO objects (id, name, color) VALUES (5, 'hi', 'yellow');", Expected: []sql.Row{{sql.NewOkResult(0)}}, }, { Query: "SELECT * FROM objects;", Expected: []sql.Row{{1, "truck", "red"}, {2, "ball", "green"}, {3, "shoe", "blue"}}, }, }, }, { Name: "Delayed foreign key resolution", SetUpScript: []string{ "SET FOREIGN_KEY_CHECKS=0;", "CREATE TABLE delayed_child (pk INT PRIMARY KEY, v1 INT, CONSTRAINT fk_delayed FOREIGN KEY (v1) REFERENCES delayed_parent(v1));", "CREATE TABLE delayed_parent (pk INT PRIMARY KEY, v1 INT, INDEX (v1));", "INSERT INTO delayed_child VALUES (1, 2);", "SET FOREIGN_KEY_CHECKS=1;", }, Assertions: []ScriptTestAssertion{ { Query: "SHOW CREATE TABLE delayed_child;", Expected: []sql.Row{{"delayed_child", "CREATE TABLE `delayed_child` (\n `pk` int NOT NULL,\n `v1` int,\n PRIMARY KEY (`pk`),\n CONSTRAINT `fk_delayed` FOREIGN KEY (`v1`) REFERENCES `delayed_parent` (`v1`)\n) ENGINE=InnoDB DEFAULT CHARSET=utf8mb4 COLLATE=utf8mb4_0900_bin"}}, }, { Query: "SELECT * FROM delayed_parent;", Expected: []sql.Row{}, }, { Query: "SELECT * FROM delayed_child;", Expected: []sql.Row{{1, 2}}, }, { Query: "INSERT INTO delayed_child VALUES (2, 3);", ExpectedErr: sql.ErrForeignKeyNotResolved, }, { Query: "INSERT INTO delayed_parent VALUES (1, 2), (2, 3);", Expected: []sql.Row{{sql.NewOkResult(2)}}, }, { Query: "INSERT INTO delayed_child VALUES (2, 3);", Expected: []sql.Row{{sql.NewOkResult(1)}}, }, { Query: "SELECT * FROM delayed_child;", Expected: []sql.Row{{1, 2}, {2, 3}}, }, }, }, { Name: "Delayed foreign key resolution resetting FOREIGN_KEY_CHECKS", SetUpScript: []string{ "SET FOREIGN_KEY_CHECKS=0;", "CREATE TABLE delayed_child (pk INT PRIMARY KEY, v1 INT, CONSTRAINT fk_delayed FOREIGN KEY (v1) REFERENCES delayed_parent(v1));", "INSERT INTO delayed_child VALUES (1, 2);", "SET FOREIGN_KEY_CHECKS=1;", }, Assertions: []ScriptTestAssertion{ { Query: "SHOW CREATE TABLE delayed_child;", Expected: []sql.Row{{"delayed_child", "CREATE TABLE `delayed_child` (\n `pk` int NOT NULL,\n `v1` int,\n PRIMARY KEY (`pk`),\n CONSTRAINT `fk_delayed` FOREIGN KEY (`v1`) REFERENCES `delayed_parent` (`v1`)\n) ENGINE=InnoDB DEFAULT CHARSET=utf8mb4 COLLATE=utf8mb4_0900_bin"}}, }, { Query: "SELECT * FROM delayed_child;", Expected: []sql.Row{{1, 2}}, }, { Query: "INSERT INTO delayed_child VALUES (2, 3);", ExpectedErr: sql.ErrForeignKeyNotResolved, }, { Query: "CREATE TABLE delayed_parent (pk INT PRIMARY KEY, v1 INT, INDEX (v1));", Expected: []sql.Row{{sql.NewOkResult(0)}}, }, { Query: "INSERT INTO delayed_parent VALUES (1, 2), (2, 3);", Expected: []sql.Row{{sql.NewOkResult(2)}}, }, { Query: "INSERT INTO delayed_child VALUES (2, 3);", Expected: []sql.Row{{sql.NewOkResult(1)}}, }, { Query: "SELECT * FROM delayed_child;", Expected: []sql.Row{{1, 2}, {2, 3}}, }, }, }, { Name: "DROP TABLE with FOREIGN_KEY_CHECKS=0", SetUpScript: []string{ "ALTER TABLE child ADD CONSTRAINT fk_dropped FOREIGN KEY (v1) REFERENCES parent(v1);", }, Assertions: []ScriptTestAssertion{ { Query: "DROP TABLE parent;", ExpectedErr: sql.ErrForeignKeyDropTable, }, { Query: "SET FOREIGN_KEY_CHECKS=0;", Expected: []sql.Row{{}}, }, { Query: "DROP TABLE parent;", Expected: []sql.Row{{sql.NewOkResult(0)}}, }, { Query: "SET FOREIGN_KEY_CHECKS=1;", Expected: []sql.Row{{}}, }, { Query: "INSERT INTO child VALUES (4, 5, 6);", ExpectedErr: sql.ErrForeignKeyNotResolved, }, { Query: "CREATE TABLE parent (pk INT PRIMARY KEY, v1 INT, INDEX (v1));", Expected: []sql.Row{{sql.NewOkResult(0)}}, }, { Query: "INSERT INTO parent VALUES (1, 5);", Expected: []sql.Row{{sql.NewOkResult(1)}}, }, { Query: "INSERT INTO child VALUES (4, 5, 6);", Expected: []sql.Row{{sql.NewOkResult(1)}}, }, { Query: "SELECT * FROM parent;", Expected: []sql.Row{{1, 5}}, }, { Query: "SELECT * FROM child;", Expected: []sql.Row{{4, 5, 6}}, }, }, }, { Name: "ALTER TABLE ADD CONSTRAINT for different database", SetUpScript: []string{ "CREATE DATABASE public;", "CREATE TABLE public.cities (pk INT PRIMARY KEY, city VARCHAR(255), state VARCHAR(2));", "CREATE TABLE public.states (state_id INT PRIMARY KEY, state VARCHAR(2));", }, Assertions: []ScriptTestAssertion{ { Query: "ALTER TABLE public.cities ADD CONSTRAINT foreign_key1 FOREIGN KEY (state) REFERENCES public.states(state);", ExpectedErr: sql.ErrForeignKeyMissingReferenceIndex, }, { Query: "CREATE INDEX foreign_key1 ON public.states(state);", Expected: []sql.Row{{sql.NewOkResult(0)}}, }, { Query: "ALTER TABLE public.cities ADD CONSTRAINT foreign_key1 FOREIGN KEY (state) REFERENCES public.states(state);", Expected: []sql.Row{{sql.NewOkResult(0)}}, }, }, }, { Name: "Creating a foreign key on a table with an unsupported type works", SetUpScript: []string{ "CREATE TABLE IF NOT EXISTS restaurants (id INT PRIMARY KEY, coordinate POINT);", "CREATE TABLE IF NOT EXISTS hours (restaurant_id INT PRIMARY KEY AUTO_INCREMENT, CONSTRAINT fk_name FOREIGN KEY (restaurant_id) REFERENCES restaurants(id));", }, Assertions: []ScriptTestAssertion{ { Query: "SHOW CREATE TABLE hours;", Expected: []sql.Row{{"hours", "CREATE TABLE `hours` (\n `restaurant_id` int NOT NULL AUTO_INCREMENT,\n PRIMARY KEY (`restaurant_id`),\n CONSTRAINT `fk_name` FOREIGN KEY (`restaurant_id`) REFERENCES `restaurants` (`id`)\n) ENGINE=InnoDB DEFAULT CHARSET=utf8mb4 COLLATE=utf8mb4_0900_bin"}}, }, }, }, { Name: "Create foreign key onto primary key", SetUpScript: []string{ "DROP TABLE child;", "DROP TABLE parent;", "CREATE TABLE parent (a INT, b INT, c INT, PRIMARY KEY (b, a));", "CREATE TABLE child (a INT PRIMARY KEY, b INT);", }, Assertions: []ScriptTestAssertion{ { Query: "ALTER TABLE child ADD CONSTRAINT fk1 FOREIGN KEY (b) REFERENCES parent (b);", Expected: []sql.Row{{sql.NewOkResult(0)}}, }, { Query: "ALTER TABLE child ADD CONSTRAINT fk2 FOREIGN KEY (a) REFERENCES parent (b);", Expected: []sql.Row{{sql.NewOkResult(0)}}, }, { Query: "ALTER TABLE child ADD CONSTRAINT fk3 FOREIGN KEY (a, b) REFERENCES parent (a, b);", Expected: []sql.Row{{sql.NewOkResult(0)}}, }, { Query: "ALTER TABLE child ADD CONSTRAINT fk4 FOREIGN KEY (b, a) REFERENCES parent (b, a);", Expected: []sql.Row{{sql.NewOkResult(0)}}, }, }, }, { Name: "Reordered foreign key columns match an index's prefix, INSERT values", SetUpScript: []string{ "DROP TABLE child;", "DROP TABLE parent;", "CREATE TABLE parent(pk DOUBLE PRIMARY KEY, v1 BIGINT, v2 BIGINT, INDEX(v1, v2, pk));", "INSERT INTO parent VALUES (1, 1, 1), (2, 1, 2);", "CREATE TABLE child(pk BIGINT PRIMARY KEY, v1 BIGINT, v2 BIGINT, CONSTRAINT fk_child FOREIGN KEY (v2, v1) REFERENCES parent(v2, v1));", }, Assertions: []ScriptTestAssertion{ { Query: "INSERT INTO child VALUES (1, 1, 1);", Expected: []sql.Row{{sql.NewOkResult(1)}}, }, { Query: "INSERT INTO child VALUES (2, 2, 2);", ExpectedErr: sql.ErrForeignKeyChildViolation, }, { Query: "INSERT INTO child VALUES (3, 2, 1);", ExpectedErr: sql.ErrForeignKeyChildViolation, }, { Query: "INSERT INTO child VALUES (4, 1, 2);", Expected: []sql.Row{{sql.NewOkResult(1)}}, }, }, }, { Name: "Reordered foreign key columns match an index's prefix, ALTER TABLE ADD FOREIGN KEY fails check", SetUpScript: []string{ "DROP TABLE child;", "DROP TABLE parent;", "CREATE TABLE parent(pk DOUBLE PRIMARY KEY, v1 BIGINT, v2 BIGINT, INDEX(v1, v2, pk));", "INSERT INTO parent VALUES (1, 1, 1), (2, 1, 2);", "CREATE TABLE child(pk BIGINT PRIMARY KEY, v1 BIGINT, v2 BIGINT);", "INSERT INTO child VALUES (1, 2, 1);", }, Assertions: []ScriptTestAssertion{ { Query: "ALTER TABLE child ADD CONSTRAINT fk_child FOREIGN KEY (v2, v1) REFERENCES parent(v2, v1);", ExpectedErr: sql.ErrForeignKeyChildViolation, }, }, }, { Name: "Self-referential deletion with ON UPDATE CASCADE", SetUpScript: []string{ "CREATE TABLE self(pk BIGINT PRIMARY KEY, v1 BIGINT, v2 BIGINT, INDEX(v1), CONSTRAINT fk_self FOREIGN KEY(v2) REFERENCES self(v1) ON UPDATE CASCADE);", "INSERT INTO self VALUES (0, 1, 1), (1, 2, 1);", }, Assertions: []ScriptTestAssertion{ { Query: "DELETE FROM self WHERE v1 = 1;", ExpectedErr: sql.ErrForeignKeyParentViolation, }, { Query: "DELETE FROM self WHERE v1 = 2;", Expected: []sql.Row{{sql.NewOkResult(1)}}, }, }, }, { Name: "Self-referential deletion with ON DELETE CASCADE", SetUpScript: []string{ "CREATE TABLE self(pk BIGINT PRIMARY KEY, v1 BIGINT, v2 BIGINT, INDEX(v1), CONSTRAINT fk_self FOREIGN KEY(v2) REFERENCES self(v1) ON DELETE CASCADE);", "INSERT INTO self VALUES (0, 1, 1), (1, 2, 1);", }, Assertions: []ScriptTestAssertion{ { Query: "DELETE FROM self WHERE v1 = 1;", Expected: []sql.Row{{sql.NewOkResult(1)}}, }, { Query: "SELECT * FROM self;", Expected: []sql.Row{}, }, }, }, { Name: "Cascaded DELETE becomes cascading UPDATE after first child, using ON DELETE for second child", SetUpScript: []string{ "DROP TABLE child;", "DROP TABLE parent;", "CREATE TABLE parent (pk BIGINT PRIMARY KEY, v1 BIGINT, v2 BIGINT, INDEX (v1), INDEX (v2), INDEX (v1, v2));", "CREATE TABLE child (pk BIGINT PRIMARY KEY, v1 BIGINT, v2 BIGINT, CONSTRAINT fk_child FOREIGN KEY (v1, v2) REFERENCES parent (v1, v2) ON DELETE SET NULL);", "CREATE TABLE child2 (pk BIGINT PRIMARY KEY, v1 BIGINT, v2 BIGINT, CONSTRAINT fk_child2 FOREIGN KEY (v1, v2) REFERENCES child (v1, v2) ON DELETE SET NULL);", "INSERT INTO parent VALUES (1,1,1), (2,2,2), (3,3,3);", "INSERT INTO child VALUES (1,1,1), (2,2,2), (3,3,3);", "INSERT INTO child2 VALUES (1,1,1), (2,2,2), (3,3,3);", }, Assertions: []ScriptTestAssertion{ { Query: "DELETE FROM parent WHERE pk = 1;", ExpectedErr: sql.ErrForeignKeyParentViolation, }, }, }, { Name: "Cascaded DELETE becomes cascading UPDATE after first child, using ON UPDATE for second child", SetUpScript: []string{ "DROP TABLE child;", "DROP TABLE parent;", "CREATE TABLE parent (pk BIGINT PRIMARY KEY, v1 BIGINT, v2 BIGINT, INDEX (v1), INDEX (v2), INDEX (v1, v2));", "CREATE TABLE child (pk BIGINT PRIMARY KEY, v1 BIGINT, v2 BIGINT, CONSTRAINT fk_child FOREIGN KEY (v1, v2) REFERENCES parent (v1, v2) ON DELETE SET NULL);", "CREATE TABLE child2 (pk BIGINT PRIMARY KEY, v1 BIGINT, v2 BIGINT, CONSTRAINT fk_child2 FOREIGN KEY (v1, v2) REFERENCES child (v1, v2) ON UPDATE CASCADE);", "INSERT INTO parent VALUES (1,1,1), (2,2,2), (3,3,3);", "INSERT INTO child VALUES (1,1,1), (2,2,2), (3,3,3);", "INSERT INTO child2 VALUES (1,1,1), (2,2,2), (3,3,3);", }, Assertions: []ScriptTestAssertion{ { Query: "DELETE FROM parent WHERE pk = 1;", Expected: []sql.Row{{sql.NewOkResult(1)}}, }, { Query: "SELECT * FROM parent;", Expected: []sql.Row{{2, 2, 2}, {3, 3, 3}}, }, { Query: "SELECT * FROM child;", Expected: []sql.Row{{1, nil, nil}, {2, 2, 2}, {3, 3, 3}}, }, { Query: "SELECT * FROM child2;", Expected: []sql.Row{{1, nil, nil}, {2, 2, 2}, {3, 3, 3}}, }, }, }, { Name: "INSERT on DUPLICATE correctly works with FKs", SetUpScript: []string{ "INSERT INTO parent values (1,1,1),(2,2,2),(3,3,3)", "ALTER TABLE child ADD CONSTRAINT fk_named FOREIGN KEY (v1) REFERENCES parent(v1);", "INSERT into child values (1, 1, 1)", "CREATE TABLE one (pk BIGINT PRIMARY KEY, v1 BIGINT, v2 BIGINT, INDEX v1 (v1));", "CREATE TABLE two (pk BIGINT PRIMARY KEY, v1 BIGINT, v2 BIGINT, INDEX v1v2 (v1, v2), CONSTRAINT fk_name_1 FOREIGN KEY (v1) REFERENCES one(v1) ON DELETE CASCADE ON UPDATE CASCADE);", "INSERT INTO one VALUES (1, 1, 4), (2, 2, 5), (3, 3, 6), (4, 4, 5);", "INSERT INTO two VALUES (2, 1, 1), (3, 2, 2), (4, 3, 3), (5, 4, 4);", }, Assertions: []ScriptTestAssertion{ { Query: "INSERT INTO parent VALUES (1,200,1) ON DUPLICATE KEY UPDATE v1 = values(v1)", ExpectedErr: sql.ErrForeignKeyParentViolation, }, { Query: "INSERT INTO one VALUES (1, 2, 4) on duplicate key update v1 = VALUES(v1)", Expected: []sql.Row{{sql.NewOkResult(2)}}, }, { Query: "SELECT * FROM two where pk = 2", Expected: []sql.Row{{2, 2, 1}}, }, }, }, { Name: "Referencing Primary Key", SetUpScript: []string{ "CREATE table parent1 (pk BIGINT PRIMARY KEY, v1 BIGINT);", "CREATE table child1 (pk BIGINT PRIMARY KEY, v1 BIGINT, FOREIGN KEY (v1) REFERENCES parent1(pk) ON UPDATE CASCADE ON DELETE CASCADE);", "INSERT INTO parent1 VALUES (1, 1);", }, Assertions: []ScriptTestAssertion{ { Query: "INSERT INTO child1 VALUES (1, 1);", Expected: []sql.Row{{sql.NewOkResult(1)}}, }, { Query: "SELECT * FROM child1;", Expected: []sql.Row{{1, 1}}, }, { Query: "UPDATE parent1 SET pk = 2;", Expected: []sql.Row{{sql.OkResult{RowsAffected: 1, Info: plan.UpdateInfo{Matched: 1, Updated: 1}}}}, }, { Query: "SELECT * FROM child1;", Expected: []sql.Row{{1, 2}}, }, }, }, { Name: "Referencing Composite Primary Key", SetUpScript: []string{ "CREATE table parent1 (pk1 BIGINT, pk2 BIGINT, v1 BIGINT, PRIMARY KEY(pk1, pk2));", "CREATE table child1 (pk BIGINT PRIMARY KEY, v1 BIGINT, v2 BIGINT, FOREIGN KEY (v1, v2) REFERENCES parent1(pk1, pk2) ON UPDATE CASCADE ON DELETE CASCADE);", "INSERT INTO parent1 VALUES (1, 2, 3), (4, 5, 6);", }, Assertions: []ScriptTestAssertion{ { Query: "INSERT INTO child1 VALUES (1, 1, 2), (2, 4, 5);", Expected: []sql.Row{{sql.NewOkResult(2)}}, }, { Query: "SELECT * FROM child1;", Expected: []sql.Row{{1, 1, 2}, {2, 4, 5}}, }, { Query: "UPDATE parent1 SET pk2 = pk1 + pk2;", Expected: []sql.Row{{sql.OkResult{RowsAffected: 2, Info: plan.UpdateInfo{Matched: 2, Updated: 2}}}}, }, { Query: "SELECT * FROM child1;", Expected: []sql.Row{{1, 1, 3}, {2, 4, 9}}, }, }, }, { Name: "Keyless CASCADE deleting all rows", SetUpScript: []string{ "CREATE TABLE one (v0 BIGINT, v1 BIGINT, INDEX one_v0 (v0), INDEX one_v1 (v1));", "CREATE TABLE two (v1 BIGINT, CONSTRAINT fk_name_1 FOREIGN KEY (v1) REFERENCES one(v1) ON DELETE CASCADE ON UPDATE CASCADE);", "INSERT INTO one VALUES (1, 2);", "INSERT INTO two VALUES (2);", "UPDATE one SET v1 = v0 + v1;", "DELETE FROM one WHERE v0 = 1;", }, Assertions: []ScriptTestAssertion{ { Query: "SELECT * FROM one;", Expected: []sql.Row{}, }, { Query: "SELECT * FROM two;", Expected: []sql.Row{}, }, }, }, { Name: "Keyless CASCADE over three tables", SetUpScript: []string{ "CREATE TABLE one (v0 BIGINT, v1 BIGINT, v2 BIGINT, INDEX idx (v0));", "ALTER TABLE one ADD INDEX v1 (v1);", "CREATE TABLE two (v0 BIGINT, v1 BIGINT, v2 BIGINT, INDEX idx (v0), CONSTRAINT fk_name_1 FOREIGN KEY (v1) REFERENCES one(v1) ON DELETE CASCADE ON UPDATE CASCADE);", "ALTER TABLE two ADD INDEX v1v2 (v1, v2);", "CREATE TABLE three (v0 BIGINT, v1 BIGINT, v2 BIGINT, INDEX idx (v0), CONSTRAINT fk_name_2 FOREIGN KEY (v1, v2) REFERENCES two(v1, v2) ON DELETE CASCADE ON UPDATE CASCADE);", "INSERT INTO one VALUES (1, 1, 4), (2, 2, 5), (3, 3, 6), (4, 4, 5);", "INSERT INTO two VALUES (2, 1, 1), (3, 2, 2), (4, 3, 3), (5, 4, 4);", "INSERT INTO three VALUES (3, 1, 1), (4, 2, 2), (5, 3, 3), (6, 4, 4);", "UPDATE one SET v1 = v1 + v2;", "DELETE FROM one WHERE v0 = 3;", "UPDATE two SET v2 = v1 - 2;", }, Assertions: []ScriptTestAssertion{ { Query: "SELECT * FROM one;", Expected: []sql.Row{{1, 5, 4}, {2, 7, 5}, {4, 9, 5}}, }, { Query: "SELECT * FROM two;", Expected: []sql.Row{{2, 5, 3}, {3, 7, 5}}, }, { Query: "SELECT * FROM three;", Expected: []sql.Row{{3, 5, 3}, {4, 7, 5}}, }, }, }, }
ForeignKeyTests will run the following statements BEFORE the SetUpScript: CREATE TABLE parent (id INT PRIMARY KEY, v1 INT, v2 INT, INDEX v1 (v1), INDEX v2 (v2)); CREATE TABLE child (id INT PRIMARY KEY, v1 INT, v2 INT);
var GenericUpdateErrorTests = []GenericErrorQueryTest{
{
Name: "invalid table",
Query: "UPDATE doesnotexist SET i = 0;",
},
{
Name: "missing binding",
Query: "UPDATE mytable SET i = ?;",
},
{
Name: "wrong number of columns",
Query: `UPDATE mytable SET i = ("one", "two");`,
},
{
Name: "type mismatch: string -> int",
Query: `UPDATE mytable SET i = "one"`,
},
{
Name: "type mismatch: string -> float",
Query: `UPDATE floattable SET f64 = "one"`,
},
{
Name: "type mismatch: string -> uint",
Query: `UPDATE typestable SET f64 = "one"`,
},
{
Name: "invalid column set",
Query: "UPDATE mytable SET z = 0;",
},
{
Name: "invalid column set value",
Query: "UPDATE mytable SET i = z;",
},
{
Name: "invalid column where",
Query: "UPDATE mytable SET s = 'hi' WHERE z = 1;",
},
{
Name: "invalid column order by",
Query: "UPDATE mytable SET s = 'hi' ORDER BY z;",
},
{
Name: "negative limit",
Query: "UPDATE mytable SET s = 'hi' LIMIT -1;",
},
{
Name: "negative offset",
Query: "UPDATE mytable SET s = 'hi' LIMIT 1 OFFSET -1;",
},
{
Name: "set null on non-nullable",
Query: "UPDATE mytable SET s = NULL;",
},
{
Name: "targets join",
Query: "UPDATE mytable one, mytable two SET s = NULL;",
},
{
Name: "targets subquery alias",
Query: "UPDATE (SELECT * FROM mytable) mytable SET s = NULL;",
},
}
var IndexPlanTests = []QueryPlanTest{}/* 700 elements not displayed */
var InfoSchemaQueries = []QueryTest{ { Query: "SHOW TABLES", Expected: []sql.Row{ {"myview"}, {"fk_tbl"}, {"mytable"}, }, }, { Query: "SHOW FULL TABLES", Expected: []sql.Row{ {"fk_tbl", "BASE TABLE"}, {"myview", "VIEW"}, {"mytable", "BASE TABLE"}, }, }, { Query: "SHOW TABLES FROM foo", Expected: []sql.Row{ {"other_table"}, }, }, { Query: "SHOW TABLES LIKE '%table'", Expected: []sql.Row{ {"mytable"}, }, }, { Query: `SHOW COLUMNS FROM mytable`, Expected: []sql.Row{ {"i", "bigint", "NO", "PRI", "", ""}, {"s", "varchar(20)", "NO", "UNI", "", ""}, }, }, { Query: `DESCRIBE mytable`, Expected: []sql.Row{ {"i", "bigint", "NO", "PRI", "", ""}, {"s", "varchar(20)", "NO", "UNI", "", ""}, }, }, { Query: `DESC mytable`, Expected: []sql.Row{ {"i", "bigint", "NO", "PRI", "", ""}, {"s", "varchar(20)", "NO", "UNI", "", ""}, }, }, { Query: `SHOW COLUMNS FROM mytable WHERE Field = 'i'`, Expected: []sql.Row{ {"i", "bigint", "NO", "PRI", "", ""}, }, }, { Query: `SHOW COLUMNS FROM mytable LIKE 'i'`, Expected: []sql.Row{ {"i", "bigint", "NO", "PRI", "", ""}, }, }, { Query: `SHOW FULL COLUMNS FROM mytable`, Expected: []sql.Row{ {"i", "bigint", nil, "NO", "PRI", "", "", "", ""}, {"s", "varchar(20)", "utf8mb4_0900_bin", "NO", "UNI", "", "", "", "column s"}, }, }, { Query: "SHOW TABLES WHERE `Tables_in_mydb` = 'mytable'", Expected: []sql.Row{ {"mytable"}, }, }, { Query: ` SELECT LOGFILE_GROUP_NAME, FILE_NAME, TOTAL_EXTENTS, INITIAL_SIZE, ENGINE, EXTRA FROM INFORMATION_SCHEMA.FILES WHERE FILE_TYPE = 'UNDO LOG' AND FILE_NAME IS NOT NULL AND LOGFILE_GROUP_NAME IS NOT NULL GROUP BY LOGFILE_GROUP_NAME, FILE_NAME, ENGINE, TOTAL_EXTENTS, INITIAL_SIZE ORDER BY LOGFILE_GROUP_NAME `, Expected: nil, }, { Query: ` SELECT DISTINCT TABLESPACE_NAME, FILE_NAME, LOGFILE_GROUP_NAME, EXTENT_SIZE, INITIAL_SIZE, ENGINE FROM INFORMATION_SCHEMA.FILES WHERE FILE_TYPE = 'DATAFILE' ORDER BY TABLESPACE_NAME, LOGFILE_GROUP_NAME `, Expected: nil, }, { Query: ` SELECT COLUMN_NAME, JSON_EXTRACT(HISTOGRAM, '$."number-of-buckets-specified"') FROM information_schema.COLUMN_STATISTICS WHERE SCHEMA_NAME = 'mydb' AND TABLE_NAME = 'mytable' `, Expected: nil, }, { Query: ` SELECT TABLE_NAME FROM information_schema.TABLES WHERE TABLE_SCHEMA='mydb' AND (TABLE_TYPE='BASE TABLE' OR TABLE_TYPE='VIEW') ORDER BY 1 `, Expected: []sql.Row{ {"fk_tbl"}, {"mytable"}, {"myview"}, }, }, { Query: ` SELECT COLUMN_NAME, DATA_TYPE FROM information_schema.COLUMNS WHERE TABLE_SCHEMA='mydb' AND TABLE_NAME='mytable' `, Expected: []sql.Row{ {"s", "varchar"}, {"i", "bigint"}, }, }, { Query: ` SELECT COLUMN_NAME FROM information_schema.COLUMNS WHERE TABLE_SCHEMA=DATABASE() AND TABLE_NAME LIKE '%table' GROUP BY COLUMN_NAME `, Expected: []sql.Row{ {"s"}, {"i"}, }, }, { Query: ` SELECT COLUMN_NAME FROM information_schema.COLUMNS WHERE TABLE_SCHEMA=DATABASE() AND TABLE_NAME LIKE '%table' GROUP BY 1 `, Expected: []sql.Row{ {"s"}, {"i"}, }, }, { Query: ` SELECT COLUMN_NAME AS COLUMN_NAME FROM information_schema.COLUMNS WHERE TABLE_SCHEMA=DATABASE() AND TABLE_NAME LIKE '%table' GROUP BY 1 `, Expected: []sql.Row{ {"s"}, {"i"}, }, }, { Query: ` SELECT COLUMN_NAME AS COLUMN_NAME FROM information_schema.COLUMNS WHERE TABLE_SCHEMA=DATABASE() AND TABLE_NAME LIKE '%table' GROUP BY 1 HAVING SUBSTRING(COLUMN_NAME, 1, 1) = "s" `, Expected: []sql.Row{{"s"}}, }, { Query: `SHOW INDEXES FROM mytaBLE`, Expected: []sql.Row{ {"mytable", 0, "PRIMARY", 1, "i", nil, 0, nil, nil, "", "BTREE", "", "", "YES", nil}, {"mytable", 0, "mytable_s", 1, "s", nil, 0, nil, nil, "", "BTREE", "", "", "YES", nil}, {"mytable", 1, "mytable_i_s", 1, "i", nil, 0, nil, nil, "", "BTREE", "", "", "YES", nil}, {"mytable", 1, "mytable_i_s", 2, "s", nil, 0, nil, nil, "", "BTREE", "", "", "YES", nil}, {"mytable", 1, "idx_si", 1, "s", nil, 0, nil, nil, "", "BTREE", "", "", "YES", nil}, {"mytable", 1, "idx_si", 2, "i", nil, 0, nil, nil, "", "BTREE", "", "", "YES", nil}, }, }, { Query: `SHOW KEYS FROM mytaBLE`, Expected: []sql.Row{ {"mytable", 0, "PRIMARY", 1, "i", nil, 0, nil, nil, "", "BTREE", "", "", "YES", nil}, {"mytable", 0, "mytable_s", 1, "s", nil, 0, nil, nil, "", "BTREE", "", "", "YES", nil}, {"mytable", 1, "mytable_i_s", 1, "i", nil, 0, nil, nil, "", "BTREE", "", "", "YES", nil}, {"mytable", 1, "mytable_i_s", 2, "s", nil, 0, nil, nil, "", "BTREE", "", "", "YES", nil}, {"mytable", 1, "idx_si", 1, "s", nil, 0, nil, nil, "", "BTREE", "", "", "YES", nil}, {"mytable", 1, "idx_si", 2, "i", nil, 0, nil, nil, "", "BTREE", "", "", "YES", nil}, }, }, { Query: `SHOW CREATE TABLE mytaBLE`, Expected: []sql.Row{ {"mytable", "CREATE TABLE `mytable` (\n" + " `i` bigint NOT NULL,\n" + " `s` varchar(20) NOT NULL COMMENT 'column s',\n" + " PRIMARY KEY (`i`),\n" + " KEY `idx_si` (`s`,`i`),\n" + " KEY `mytable_i_s` (`i`,`s`),\n" + " UNIQUE KEY `mytable_s` (`s`)\n" + ") ENGINE=InnoDB DEFAULT CHARSET=utf8mb4 COLLATE=utf8mb4_0900_bin"}, }, }, { Query: `SHOW CREATE TABLE fk_TBL`, Expected: []sql.Row{ {"fk_tbl", "CREATE TABLE `fk_tbl` (\n" + " `pk` bigint NOT NULL,\n" + " `a` bigint,\n" + " `b` varchar(20),\n" + " PRIMARY KEY (`pk`),\n" + " KEY `ab` (`a`,`b`),\n" + " CONSTRAINT `fk1` FOREIGN KEY (`a`,`b`) REFERENCES `mytable` (`i`,`s`) ON DELETE CASCADE\n" + ") ENGINE=InnoDB DEFAULT CHARSET=utf8mb4 COLLATE=utf8mb4_0900_bin"}, }, }, { Query: "SELECT table_name, `auto_increment` FROM information_schema.tables " + "WHERE TABLE_SCHEMA='mydb' AND TABLE_TYPE='BASE TABLE' ORDER BY 1", Expected: []sql.Row{ {"fk_tbl", nil}, {"mytable", nil}, }, }, { Query: "SHOW ENGINES", Expected: []sql.Row{ {"InnoDB", "DEFAULT", "Supports transactions, row-level locking, and foreign keys", "YES", "YES", "YES"}, }, }, { Query: "SELECT * FROM information_schema.table_constraints ORDER BY table_name, constraint_type;", Expected: []sql.Row{ {"def", "mydb", "fk1", "mydb", "fk_tbl", "FOREIGN KEY", "YES"}, {"def", "mydb", "PRIMARY", "mydb", "fk_tbl", "PRIMARY KEY", "YES"}, {"def", "mydb", "PRIMARY", "mydb", "mytable", "PRIMARY KEY", "YES"}, {"def", "mydb", "mytable_s", "mydb", "mytable", "UNIQUE", "YES"}, {"def", "foo", "PRIMARY", "foo", "other_table", "PRIMARY KEY", "YES"}, }, }, { Query: "SELECT * FROM information_schema.check_constraints ORDER BY constraint_schema, constraint_name, check_clause ", Expected: []sql.Row{}, }, { Query: "SELECT * FROM information_schema.key_column_usage ORDER BY constraint_schema, table_name", Expected: []sql.Row{ {"def", "foo", "PRIMARY", "def", "foo", "other_table", "text", 1, nil, nil, nil, nil}, {"def", "mydb", "PRIMARY", "def", "mydb", "fk_tbl", "pk", 1, nil, nil, nil, nil}, {"def", "mydb", "fk1", "def", "mydb", "fk_tbl", "a", 1, 1, "mydb", "mytable", "i"}, {"def", "mydb", "fk1", "def", "mydb", "fk_tbl", "b", 2, 2, "mydb", "mytable", "s"}, {"def", "mydb", "PRIMARY", "def", "mydb", "mytable", "i", 1, nil, nil, nil, nil}, {"def", "mydb", "mytable_s", "def", "mydb", "mytable", "s", 1, nil, nil, nil, nil}, }, }, { Query: "SELECT * FROM information_schema.partitions", Expected: []sql.Row{}, }, { Query: ` select CONCAT(tbl.table_schema, '.', tbl.table_name) as the_table, col.column_name, GROUP_CONCAT(kcu.column_name SEPARATOR ',') as pk from information_schema.tables as tbl join information_schema.columns as col on tbl.table_name = col.table_name join information_schema.key_column_usage as kcu on tbl.table_name = kcu.table_name join information_schema.table_constraints as tc on kcu.constraint_name = tc.constraint_name where tbl.table_schema = 'mydb' and tbl.table_name = kcu.table_name and tc.constraint_type = 'PRIMARY KEY' and col.column_name like 'pk%' group by the_table, col.column_name `, Expected: []sql.Row{ {"mydb.fk_tbl", "pk", "pk,pk,pk"}, }, }, { Query: `SELECT * FROM information_schema.COLLATION_CHARACTER_SET_APPLICABILITY ORDER BY collation_name LIMIT 4 `, Expected: []sql.Row{ {"armscii8_bin", "armscii8"}, {"armscii8_general_ci", "armscii8"}, {"ascii_bin", "ascii"}, {"ascii_general_ci", "ascii"}, }, }, { Query: `SELECT * from information_schema.administrable_role_authorizations`, Expected: []sql.Row{}, }, { Query: `SELECT * from information_schema.applicable_roles`, Expected: []sql.Row{}, }, { Query: `SELECT * FROM information_schema.column_privileges`, Expected: []sql.Row{}, }, { Query: `SELECT * FROM information_schema.columns_extensions`, Expected: []sql.Row{}, }, { Query: `SELECT * FROM information_schema.connection_control_failed_login_attempts`, Expected: []sql.Row{}, }, { Query: `SELECT * FROM information_schema.keywords`, Expected: []sql.Row{}, }, { Query: `SELECT * FROM information_schema.mysql_firewall_users`, Expected: []sql.Row{}, }, { Query: `SELECT * FROM information_schema.mysql_firewall_whitelist`, Expected: []sql.Row{}, }, { Query: `SELECT * FROM information_schema.optimizer_trace`, Expected: []sql.Row{}, }, { Query: `SELECT * FROM information_schema.plugins`, Expected: []sql.Row{}, }, { Query: `SELECT * FROM information_schema.profiling`, Expected: []sql.Row{}, }, { Query: `SELECT * FROM information_schema.resource_groups`, Expected: []sql.Row{}, }, { Query: `SELECT * FROM information_schema.role_column_grants`, Expected: []sql.Row{}, }, { Query: `SELECT * FROM information_schema.role_routine_grants`, Expected: []sql.Row{}, }, { Query: `SELECT * FROM information_schema.schema_privileges`, Expected: []sql.Row{}, }, { Query: `SELECT * FROM information_schema.schemata_extensions`, Expected: []sql.Row{}, }, { Query: `SELECT * FROM information_schema.st_geometry_columns`, Expected: []sql.Row{}, }, { Query: `SELECT * FROM information_schema.st_spatial_reference_systems`, Expected: []sql.Row{}, }, { Query: `SELECT * FROM information_schema.st_units_of_measure`, Expected: []sql.Row{}, }, { Query: `SELECT * FROM information_schema.table_constraints_extensions`, Expected: []sql.Row{}, }, { Query: `SELECT * FROM information_schema.table_privileges`, Expected: []sql.Row{}, }, { Query: `SELECT * FROM information_schema.tables_extensions`, Expected: []sql.Row{}, }, { Query: `SELECT * FROM information_schema.tablespaces`, Expected: []sql.Row{}, }, { Query: `SELECT * FROM information_schema.tablespaces_extensions`, Expected: []sql.Row{}, }, { Query: `SELECT * FROM information_schema.user_attributes`, Expected: []sql.Row{}, }, { Query: `SELECT * FROM information_schema.view_routine_usage`, Expected: []sql.Row{}, }, { Query: `SELECT * FROM information_schema.view_table_usage`, Expected: []sql.Row{}, }, { Query: `SELECT * from information_schema.innodb_buffer_page`, Expected: []sql.Row{}, }, { Query: `SELECT * from information_schema.innodb_buffer_page_lru`, Expected: []sql.Row{}, }, { Query: `SELECT * from information_schema.innodb_buffer_pool_stats`, Expected: []sql.Row{}, }, { Query: `SELECT * from information_schema.innodb_cached_indexes`, Expected: []sql.Row{}, }, { Query: `SELECT * from information_schema.innodb_cmp`, Expected: []sql.Row{}, }, { Query: `SELECT * from information_schema.innodb_cmp_reset`, Expected: []sql.Row{}, }, { Query: `SELECT * from information_schema.innodb_cmpmem`, Expected: []sql.Row{}, }, { Query: `SELECT * from information_schema.innodb_cmpmem_reset`, Expected: []sql.Row{}, }, { Query: `SELECT * from information_schema.innodb_cmp_per_index`, Expected: []sql.Row{}, }, { Query: `SELECT * from information_schema.innodb_cmp_per_index_reset`, Expected: []sql.Row{}, }, { Query: `SELECT * from information_schema.innodb_columns`, Expected: []sql.Row{}, }, { Query: `SELECT * from information_schema.innodb_datafiles`, Expected: []sql.Row{}, }, { Query: `SELECT * from information_schema.innodb_fields`, Expected: []sql.Row{}, }, { Query: `SELECT * from information_schema.innodb_foreign`, Expected: []sql.Row{}, }, { Query: `SELECT * from information_schema.innodb_foreign_cols`, Expected: []sql.Row{}, }, { Query: `SELECT * from information_schema.innodb_ft_being_deleted`, Expected: []sql.Row{}, }, { Query: `SELECT * from information_schema.innodb_ft_config`, Expected: []sql.Row{}, }, { Query: `SELECT * from information_schema.innodb_ft_default_stopword`, Expected: []sql.Row{}, }, { Query: `SELECT * from information_schema.innodb_ft_deleted`, Expected: []sql.Row{}, }, { Query: `SELECT * from information_schema.innodb_ft_index_cache`, Expected: []sql.Row{}, }, { Query: `SELECT * from information_schema.innodb_ft_index_table`, Expected: []sql.Row{}, }, { Query: `SELECT * from information_schema.innodb_indexes`, Expected: []sql.Row{}, }, { Query: `SELECT * from information_schema.innodb_metrics`, Expected: []sql.Row{}, }, { Query: `SELECT * from information_schema.innodb_session_temp_tablespaces`, Expected: []sql.Row{}, }, { Query: `SELECT * from information_schema.innodb_tables`, Expected: []sql.Row{}, }, { Query: `SELECT * from information_schema.innodb_tablespaces`, Expected: []sql.Row{}, }, { Query: `SELECT * from information_schema.innodb_tablespaces_brief`, Expected: []sql.Row{}, }, { Query: `SELECT * from information_schema.innodb_tablestats`, Expected: []sql.Row{}, }, { Query: `SELECT * from information_schema.innodb_temp_table_info`, Expected: []sql.Row{}, }, { Query: `SELECT * from information_schema.innodb_trx`, Expected: []sql.Row{}, }, { Query: `SELECT * from information_schema.innodb_virtual`, Expected: []sql.Row{}, }, }
var InfoSchemaScripts = []ScriptTest{ { Name: "information_schema.table_constraints ignores non-unique indexes", SetUpScript: []string{ "CREATE TABLE t (pk int primary key, test_score int, height int)", "CREATE INDEX myindex on t(test_score)", }, Assertions: []ScriptTestAssertion{ { Query: "SELECT * FROM information_schema.table_constraints where table_name='t' ORDER BY constraint_type,constraint_name", Expected: []sql.Row{ {"def", "mydb", "PRIMARY", "mydb", "t", "PRIMARY KEY", "YES"}, }, }, }, }, { Name: "information_schema.key_column_usage ignores non-unique indexes", SetUpScript: []string{ "CREATE TABLE t (pk int primary key, test_score int, height int)", "CREATE INDEX myindex on t(test_score)", }, Assertions: []ScriptTestAssertion{ { Query: "SELECT * FROM information_schema.key_column_usage where table_name='t'", Expected: []sql.Row{ {"def", "mydb", "PRIMARY", "def", "mydb", "t", "pk", 1, nil, nil, nil, nil}, }, }, }, }, { Name: "information_schema.key_column_usage works with composite foreign keys", SetUpScript: []string{ "CREATE TABLE ptable (pk int primary key, test_score int, height int)", "CREATE INDEX myindex on ptable(test_score, height)", "CREATE TABLE ptable2 (pk int primary key, test_score2 int, height2 int, CONSTRAINT fkr FOREIGN KEY (test_score2, height2) REFERENCES ptable(test_score,height));", }, Assertions: []ScriptTestAssertion{ { Query: "SELECT * FROM information_schema.key_column_usage where table_name='ptable2' ORDER BY constraint_name", Expected: []sql.Row{ {"def", "mydb", "PRIMARY", "def", "mydb", "ptable2", "pk", 1, nil, nil, nil, nil}, {"def", "mydb", "fkr", "def", "mydb", "ptable2", "test_score2", 1, 1, "mydb", "ptable", "test_score"}, {"def", "mydb", "fkr", "def", "mydb", "ptable2", "height2", 2, 2, "mydb", "ptable", "height"}, }, }, }, }, { Name: "information_schema.key_column_usage works with composite primary keys", SetUpScript: []string{ "CREATE TABLE ptable (pk int, test_score int, height int, PRIMARY KEY (pk, test_score))", }, Assertions: []ScriptTestAssertion{ { Query: "SELECT * FROM information_schema.key_column_usage where table_name='ptable' ORDER BY constraint_name", Expected: []sql.Row{ {"def", "mydb", "PRIMARY", "def", "mydb", "ptable", "pk", 1, nil, nil, nil, nil}, {"def", "mydb", "PRIMARY", "def", "mydb", "ptable", "test_score", 2, nil, nil, nil, nil}, }, }, }, }, { Name: "information_schema.triggers create trigger definer defined", SetUpScript: []string{ "CREATE TABLE aa (x INT PRIMARY KEY)", "CREATE DEFINER=`dolt`@`localhost` TRIGGER trigger1 BEFORE INSERT ON aa FOR EACH ROW SET NEW.x = NEW.x + 1", }, Assertions: []ScriptTestAssertion{ { Query: "SELECT trigger_name, event_object_table, definer FROM INFORMATION_SCHEMA.TRIGGERS WHERE trigger_name = 'trigger1'", Expected: []sql.Row{ {"trigger1", "aa", "`dolt`@`localhost`"}, }, }, }, }, { Name: "information_schema.statistics shows non unique index", SetUpScript: []string{ "CREATE TABLE t (pk int primary key, test_score int, height int)", "CREATE INDEX myindex on t(test_score)", "INSERT INTO t VALUES (2,23,25), (3,24,26)", }, Assertions: []ScriptTestAssertion{ { Query: "SELECT * FROM information_schema.statistics where table_name='t'", Expected: []sql.Row{ {"def", "mydb", "t", 1, "mydb", "myindex", 1, "test_score", "A", int64(2), nil, nil, "YES", "BTREE", "", "", "YES", nil}, {"def", "mydb", "t", 0, "mydb", "PRIMARY", 1, "pk", "A", int64(2), nil, nil, "", "BTREE", "", "", "YES", nil}, }, }, }, }, { Name: "information_schema.columns shows default value", SetUpScript: []string{ "CREATE TABLE t (pk int primary key, fname varchar(20), lname varchar(20), height int)", "ALTER TABLE t CHANGE fname fname varchar(20) NOT NULL DEFAULT ''", "ALTER TABLE t CHANGE lname lname varchar(20) NOT NULL DEFAULT 'ln'", "ALTER TABLE t CHANGE height h int DEFAULT NULL", }, Assertions: []ScriptTestAssertion{ { Query: "SELECT table_name, column_name, column_default, is_nullable FROM information_schema.columns where table_name='t'", Expected: []sql.Row{ {"t", "pk", nil, "NO"}, {"t", "fname", "", "NO"}, {"t", "lname", "ln", "NO"}, {"t", "h", nil, "YES"}, }, }, }, }, { Name: "information_schema.columns shows default value with more types", SetUpScript: []string{ "CREATE TABLE test_table (pk int primary key, fname varchar(20), lname varchar(20), height int)", "ALTER TABLE test_table CHANGE fname col2 float NOT NULL DEFAULT 4.5", "ALTER TABLE test_table CHANGE lname col3 double NOT NULL DEFAULT 3.14159", "ALTER TABLE test_table CHANGE height col4 datetime NULL DEFAULT '2008-04-22 16:16:16'", "ALTER TABLE test_table ADD COLUMN col5 boolean NULL DEFAULT FALSE", }, Assertions: []ScriptTestAssertion{ { Query: "SELECT table_name, column_name, column_default, is_nullable FROM information_schema.CoLuMnS where table_name='test_table'", Expected: []sql.Row{ {"test_table", "pk", nil, "NO"}, {"test_table", "col2", "4.5", "NO"}, {"test_table", "col3", "3.14159", "NO"}, {"test_table", "col4", "2008-04-22 16:16:16", "YES"}, {"test_table", "col5", "false", "YES"}, }, }, }, }, { Name: "information_schema.columns shows default value with more types", SetUpScript: []string{ "CREATE TABLE test_table (pk int primary key)", "ALTER TABLE test_table ADD COLUMN col2 float DEFAULT length('hello')", "ALTER TABLE test_table ADD COLUMN col3 int DEFAULT greatest(`pk`, 2)", "ALTER TABLE test_table ADD COLUMN col4 int DEFAULT (5 + 5)", }, Assertions: []ScriptTestAssertion{ { Query: "SELECT table_name, column_name, column_default, is_nullable FROM information_schema.columns where table_name='test_table'", Expected: []sql.Row{ {"test_table", "pk", nil, "NO"}, {"test_table", "col2", "LENGTH(\"hello\")", "YES"}, {"test_table", "col3", "GREATEST(pk, 2)", "YES"}, {"test_table", "col4", "(5 + 5)", "YES"}, }, }, }, }, { Name: "information_schema.routines", SetUpScript: []string{ "CREATE PROCEDURE p1() COMMENT 'hi' DETERMINISTIC SELECT 6", "CREATE definer=`user` PROCEDURE p2() SQL SECURITY INVOKER SELECT 7", "CREATE PROCEDURE p21() SQL SECURITY DEFINER SELECT 8", "USE foo", "CREATE PROCEDURE p12() COMMENT 'hello' DETERMINISTIC SELECT 6", }, Assertions: []ScriptTestAssertion{ { Query: "SELECT specific_name, routine_catalog, routine_schema, routine_name, routine_type, " + "data_type, character_maximum_length, character_octet_length, numeric_precision, numeric_scale, " + "datetime_precision, character_set_name, collation_name, dtd_identifier, " + "routine_body, external_name, external_language, parameter_style, is_deterministic, " + "sql_data_access, sql_path, security_type, sql_mode, routine_comment, definer, " + "character_set_client, collation_connection, database_collation FROM information_schema.routines", Expected: []sql.Row{ {"p1", "def", "mydb", "p1", "PROCEDURE", "", nil, nil, nil, nil, nil, nil, nil, "", "SQL", nil, "SQL", "SQL", "", "", nil, "DEFINER", "SQL", "hi", "", "utf8mb4", "utf8mb4_0900_bin", "utf8mb4_0900_bin"}, {"p2", "def", "mydb", "p2", "PROCEDURE", "", nil, nil, nil, nil, nil, nil, nil, "", "SQL", nil, "SQL", "SQL", "", "", nil, "INVOKER", "SQL", "", "`user`@`%`", "utf8mb4", "utf8mb4_0900_bin", "utf8mb4_0900_bin"}, {"p12", "def", "foo", "p12", "PROCEDURE", "", nil, nil, nil, nil, nil, nil, nil, "", "SQL", nil, "SQL", "SQL", "", "", nil, "DEFINER", "SQL", "hello", "", "utf8mb4", "utf8mb4_0900_bin", "utf8mb4_0900_bin"}, {"p21", "def", "mydb", "p21", "PROCEDURE", "", nil, nil, nil, nil, nil, nil, nil, "", "SQL", nil, "SQL", "SQL", "", "", nil, "DEFINER", "SQL", "", "", "utf8mb4", "utf8mb4_0900_bin", "utf8mb4_0900_bin"}, }, }, }, }, { Name: "information_schema.columns for view", SetUpScript: []string{ "USE foo", "drop table other_table", "CREATE TABLE t (i int)", "CREATE VIEW v as select * from t", }, Assertions: []ScriptTestAssertion{ { Query: "SELECT * FROM INFORMATION_SCHEMA.COLUMNS WHERE table_schema = 'foo'", Expected: []sql.Row{ {"def", "foo", "t", "i", uint64(1), nil, "YES", "int", nil, nil, nil, nil, nil, nil, nil, "int", "", "", "select", "", "", "NULL"}, {"def", "foo", "v", "", uint64(0), nil, nil, nil, nil, nil, nil, nil, nil, "", "", "", "", "", "select", "", "", "NULL"}, }, }, }, }, { Name: "information_schema.columns with column key check for PRI and UNI", SetUpScript: []string{ "CREATE TABLE about (id int unsigned NOT NULL AUTO_INCREMENT, uuid char(36) NOT NULL, " + "status varchar(255) NOT NULL DEFAULT 'draft', date_created timestamp DEFAULT NULL, date_updated timestamp DEFAULT NULL, " + "url_key varchar(255) NOT NULL, PRIMARY KEY (uuid), UNIQUE KEY about_url_key_unique (url_key), UNIQUE KEY id (id))", }, Assertions: []ScriptTestAssertion{ { Query: "SELECT TABLE_NAME, COLUMN_NAME, COLUMN_DEFAULT, IS_NULLABLE, COLUMN_TYPE, COLUMN_KEY, CHARACTER_MAXIMUM_LENGTH, EXTRA FROM INFORMATION_SCHEMA.COLUMNS WHERE TABLE_NAME = 'about'", Expected: []sql.Row{ {"about", "id", nil, "NO", "int unsigned", "UNI", nil, "auto_increment"}, {"about", "uuid", nil, "NO", "char(36)", "PRI", 36, ""}, {"about", "status", "draft", "NO", "varchar(255)", "", 255, ""}, {"about", "date_created", nil, "YES", "timestamp", "", nil, ""}, {"about", "date_updated", nil, "YES", "timestamp", "", nil, ""}, {"about", "url_key", nil, "NO", "varchar(255)", "UNI", 255, ""}, }, }, }, }, { Name: "information_schema.columns with column key check for MUL", SetUpScript: []string{ "create table new_table (id int, name varchar(30), cname varbinary(100));", "alter table new_table modify column id int NOT NULL, add key(id);", }, Assertions: []ScriptTestAssertion{ { Query: "SELECT TABLE_NAME, COLUMN_NAME, IS_NULLABLE, DATA_TYPE, COLUMN_TYPE, COLUMN_KEY, CHARACTER_MAXIMUM_LENGTH, EXTRA FROM INFORMATION_SCHEMA.COLUMNS WHERE TABLE_NAME = 'new_table'", Expected: []sql.Row{ {"new_table", "id", "NO", "int", "int", "MUL", nil, ""}, {"new_table", "name", "YES", "varchar", "varchar(30)", "", 30, ""}, {"new_table", "cname", "YES", "varbinary", "varbinary(100)", "", 100, ""}, }, }, }, }, { Name: "information_schema.columns with column key check for MUL for only the first column of composite unique key", SetUpScript: []string{ "create table comp_uni (pk int not null, c0 int, c1 int, primary key (pk), unique key c0c1 (c0, c1));", }, Assertions: []ScriptTestAssertion{ { Query: "SELECT TABLE_NAME, COLUMN_NAME, IS_NULLABLE, COLUMN_TYPE, COLUMN_KEY FROM INFORMATION_SCHEMA.COLUMNS WHERE TABLE_NAME = 'comp_uni'", Expected: []sql.Row{ {"comp_uni", "pk", "NO", "int", "PRI"}, {"comp_uni", "c0", "YES", "int", "MUL"}, {"comp_uni", "c1", "YES", "int", ""}, }, }, }, }, { Name: "information_schema.columns with column key UNI is displayed as PRI if it cannot contain NULL values and there is no PRIMARY KEY in the table", SetUpScript: []string{ "create table ptable (id int not null, id2 int not null, col1 bool, UNIQUE KEY unique_key (id), UNIQUE KEY unique_key2 (id2));", }, Assertions: []ScriptTestAssertion{ { Query: "SELECT TABLE_NAME, COLUMN_NAME, IS_NULLABLE, DATA_TYPE, COLUMN_TYPE, COLUMN_KEY FROM INFORMATION_SCHEMA.COLUMNS WHERE TABLE_NAME = 'ptable'", Expected: []sql.Row{ {"ptable", "id", "NO", "int", "int", "PRI"}, {"ptable", "id2", "NO", "int", "int", "UNI"}, {"ptable", "col1", "YES", "tinyint", "tinyint(1)", ""}, }, }, }, }, { Name: "information_schema.columns with srs_id defined in spatial columns", SetUpScript: []string{ "CREATE TABLE stable (geo GEOMETRY NOT NULL DEFAULT (POINT(2, 5)), line LINESTRING NOT NULL, pnt POINT SRID 4326, pol POLYGON NOT NULL SRID 0);", }, Assertions: []ScriptTestAssertion{ { Query: "SELECT TABLE_NAME, COLUMN_NAME, COLUMN_DEFAULT, IS_NULLABLE, DATA_TYPE, COLUMN_TYPE, COLUMN_KEY, SRS_ID FROM INFORMATION_SCHEMA.COLUMNS WHERE TABLE_NAME = 'stable'", Expected: []sql.Row{ {"stable", "geo", "POINT(2, 5)", "NO", "geometry", "geometry", "", "NULL"}, {"stable", "line", nil, "NO", "linestring", "linestring", "", "NULL"}, {"stable", "pnt", nil, "YES", "point", "point", "", "4326"}, {"stable", "pol", nil, "NO", "polygon", "polygon", "", "0"}, }, }, }, }, }
var InsertBrokenScripts = []ScriptTest{ { Name: "Test that INSERT IGNORE INTO works with unique keys", SetUpScript: []string{ "CREATE TABLE mytable(pk int PRIMARY KEY, value varchar(10) UNIQUE)", "INSERT INTO mytable values (1,'one')", }, Assertions: []ScriptTestAssertion{ { Query: "INSERT IGNORE INTO mytable VALUES (2, 'one')", Expected: []sql.Row{ {sql.OkResult{RowsAffected: 0}}, }, ExpectedWarning: mysql.ERDupEntry, }, }, }, { Name: "Test that INSERT IGNORE works with FK Violations", SetUpScript: []string{ "CREATE TABLE t1 (id INT PRIMARY KEY, v int);", "CREATE TABLE t2 (id INT PRIMARY KEY, v2 int, CONSTRAINT mfk FOREIGN KEY (v2) REFERENCES t1(id));", "INSERT INTO t1 values (1,1)", }, Assertions: []ScriptTestAssertion{ { Query: "INSERT IGNORE INTO t2 VALUES (1,2);", Expected: []sql.Row{ {sql.OkResult{RowsAffected: 0}}, }, ExpectedWarning: mysql.ErNoReferencedRow2, }, }, }, { Name: "Test that INSERT IGNORE assigns the closest dataype correctly", SetUpScript: []string{ "CREATE TABLE x (pk int primary key, c1 varchar(20) NOT NULL);", `INSERT IGNORE INTO x VALUES (1, "one"), (2, TRUE), (3, "three")`, "CREATE TABLE y (pk int primary key, c1 int NOT NULL);", `INSERT IGNORE INTO y VALUES (1, 1), (2, "two"), (3,3);`, }, Assertions: []ScriptTestAssertion{ { Query: "SELECT * FROM x", Expected: []sql.Row{ {1, "one"}, {2, 1}, {3, "three"}, }, }, { Query: "SELECT * FROM y", Expected: []sql.Row{ {1, 1}, {2, 0}, {3, 3}, }, }, { Query: `INSERT IGNORE INTO y VALUES (4, "four")`, Expected: []sql.Row{ {sql.OkResult{RowsAffected: 1}}, }, ExpectedWarning: mysql.ERTruncatedWrongValueForField, }, }, }, }
var InsertErrorScripts = []ScriptTest{ { Name: "create table with non-pk auto_increment column", Query: "create table bad (pk int primary key, c0 int auto_increment);", ExpectedErr: sql.ErrInvalidAutoIncCols, }, { Name: "create multiple auto_increment columns", Query: "create table bad (pk1 int auto_increment, pk2 int auto_increment, primary key (pk1,pk2));", ExpectedErr: sql.ErrInvalidAutoIncCols, }, { Name: "create auto_increment column with default", Query: "create table bad (pk1 int auto_increment default 10, c0 int);", ExpectedErr: sql.ErrInvalidAutoIncCols, }, { Name: "try inserting string that is too long", SetUpScript: []string{ "create table bad (s varchar(9))", }, Query: "insert into bad values ('1234567890')", ExpectedErr: sql.ErrLengthBeyondLimit, }, }
var InsertErrorTests = []GenericErrorQueryTest{
{
Name: "try to insert empty into col without default value",
Query: "INSERT INTO mytable VALUES ();",
},
{
Name: "try to insert empty into col without default value",
Query: "INSERT INTO mytable () VALUES ();",
},
{
Name: "too few values",
Query: "INSERT INTO mytable (s, i) VALUES ('x');",
},
{
Name: "too many values one column",
Query: "INSERT INTO mytable (s) VALUES ('x', 999);",
},
{
Name: "missing binding",
Query: "INSERT INTO mytable (s) VALUES (?);",
},
{
Name: "too many values two columns",
Query: "INSERT INTO mytable (i, s) VALUES (999, 'x', 'y');",
},
{
Name: "too few values no columns specified",
Query: "INSERT INTO mytable VALUES (999);",
},
{
Name: "too many values no columns specified",
Query: "INSERT INTO mytable VALUES (999, 'x', 'y');",
},
{
Name: "non-existent column values",
Query: "INSERT INTO mytable (i, s, z) VALUES (999, 'x', 999);",
},
{
Name: "non-existent column set",
Query: "INSERT INTO mytable SET i = 999, s = 'x', z = 999;",
},
{
Name: "duplicate column",
Query: "INSERT INTO mytable (i, s, s) VALUES (999, 'x', 'x');",
},
{
Name: "duplicate column set",
Query: "INSERT INTO mytable SET i = 999, s = 'y', s = 'y';",
},
{
Name: "null given to non-nullable",
Query: "INSERT INTO mytable (i, s) VALUES (null, 'y');",
},
{
Name: "incompatible types",
Query: "INSERT INTO mytable (i, s) select * FROM othertable",
},
{
Name: "column count mismatch in select",
Query: "INSERT INTO mytable (i) select * FROM othertable",
},
{
Name: "column count mismatch in select",
Query: "INSERT INTO mytable select s FROM othertable",
},
{
Name: "column count mismatch in join select",
Query: "INSERT INTO mytable (s,i) SELECT * FROM othertable o JOIN mytable m ON m.i=o.i2",
},
{
Name: "duplicate key",
Query: "INSERT INTO mytable (i,s) values (1, 'hello')",
},
{
Name: "duplicate keys",
Query: "INSERT INTO mytable SELECT * from mytable",
},
{
Name: "bad column in on duplicate key update clause",
Query: "INSERT INTO mytable values (10, 'b') ON DUPLICATE KEY UPDATE notExist = 1",
},
}
var InsertIgnoreScripts = []ScriptTest{ { Name: "Test that INSERT IGNORE with Non nullable columns works", SetUpScript: []string{ "CREATE TABLE x (pk int primary key, c1 varchar(20) NOT NULL);", "INSERT IGNORE INTO x VALUES (1, NULL)", "CREATE TABLE y (pk int primary key, c1 int NOT NULL);", "INSERT IGNORE INTO y VALUES (1, NULL);", }, Assertions: []ScriptTestAssertion{ { Query: "SELECT * FROM x", Expected: []sql.Row{ {1, ""}, }, }, { Query: "SELECT * FROM y", Expected: []sql.Row{ {1, 0}, }, }, { Query: "INSERT IGNORE INTO y VALUES (2, NULL)", Expected: []sql.Row{ {sql.OkResult{RowsAffected: 1}}, }, ExpectedWarning: mysql.ERBadNullError, }, }, }, { Name: "Test that INSERT IGNORE properly addresses data conversion", SetUpScript: []string{ "CREATE TABLE t1 (pk int primary key, v1 int)", "CREATE TABLE t2 (pk int primary key, v2 varchar(1))", }, Assertions: []ScriptTestAssertion{ { Query: "INSERT IGNORE INTO t1 VALUES (1, 'dasd')", Expected: []sql.Row{ {sql.OkResult{RowsAffected: 1}}, }, ExpectedWarning: mysql.ERTruncatedWrongValueForField, }, { Query: "SELECT * FROM t1", Expected: []sql.Row{ {1, 0}, }, }, { Query: "INSERT IGNORE INTO t2 values (1, 'adsda')", Expected: []sql.Row{ {sql.OkResult{RowsAffected: 1}}, }, ExpectedWarning: mysql.ERUnknownError, }, { Query: "SELECT * FROM t2", Expected: []sql.Row{ {1, "a"}, }, }, }, }, { Name: "Insert Ignore works correctly with ON DUPLICATE UPDATE", SetUpScript: []string{ "CREATE TABLE t1 (id INT PRIMARY KEY, v int);", "INSERT INTO t1 VALUES (1,1)", "CREATE TABLE t2 (pk int primary key, v2 varchar(1))", "ALTER TABLE t2 ADD CONSTRAINT cx CHECK (pk < 100)", }, Assertions: []ScriptTestAssertion{ { Query: "INSERT IGNORE INTO t1 VALUES (1,2) ON DUPLICATE KEY UPDATE v='dsd';", Expected: []sql.Row{ {sql.OkResult{RowsAffected: 2}}, }, ExpectedWarning: mysql.ERTruncatedWrongValueForField, }, { Query: "SELECT * FROM t1", Expected: []sql.Row{ {1, 0}, }, }, { Query: "INSERT IGNORE INTO t2 values (1, 'adsda')", Expected: []sql.Row{ {sql.OkResult{RowsAffected: 1}}, }, ExpectedWarning: mysql.ERUnknownError, }, { Query: "SELECT * FROM t2", Expected: []sql.Row{ {1, "a"}, }, }, { Query: "INSERT IGNORE INTO t2 VALUES (1, 's') ON DUPLICATE KEY UPDATE pk = 1000", Expected: []sql.Row{{sql.OkResult{RowsAffected: 0}}}, }, { Query: "SELECT * FROM t2", Expected: []sql.Row{ {1, "a"}, }, }, }, }, }
var InsertIntoKeylessUnique = []WriteQueryTest{ { WriteQuery: "INSERT INTO unique_keyless VALUES (3, 3);", ExpectedWriteResult: []sql.Row{{sql.NewOkResult(1)}}, SelectQuery: "SELECT * FROM unique_keyless order by c0;", ExpectedSelect: []sql.Row{{0, 0}, {1, 1}, {2, 2}, {3, 3}}, }, { WriteQuery: "INSERT INTO unique_keyless VALUES (3, 4);", ExpectedWriteResult: []sql.Row{{sql.NewOkResult(1)}}, SelectQuery: "SELECT * FROM unique_keyless order by c0;", ExpectedSelect: []sql.Row{{0, 0}, {1, 1}, {2, 2}, {3, 4}}, }, }
var InsertIntoKeylessUniqueError = []GenericErrorQueryTest{
{
Name: "Try to insert into a unique keyless table",
Query: "INSERT INTO unique_keyless (100, 2)",
},
{
Name: "Try to insert into a unique keyless table",
Query: "INSERT INTO unique_keyless (1, 1)",
},
}
var InsertQueries = []WriteQueryTest{ { WriteQuery: "INSERT INTO keyless VALUES ();", ExpectedWriteResult: []sql.Row{{sql.NewOkResult(1)}}, SelectQuery: "SELECT * FROM keyless WHERE c0 IS NULL;", ExpectedSelect: []sql.Row{{nil, nil}}, }, { WriteQuery: "INSERT INTO keyless () VALUES ();", ExpectedWriteResult: []sql.Row{{sql.NewOkResult(1)}}, SelectQuery: "SELECT * FROM keyless WHERE c0 IS NULL;", ExpectedSelect: []sql.Row{{nil, nil}}, }, { WriteQuery: "INSERT INTO mytable (s, i) VALUES ('x', '10.0');", ExpectedWriteResult: []sql.Row{{sql.NewOkResult(1)}}, SelectQuery: "SELECT i FROM mytable WHERE s = 'x';", ExpectedSelect: []sql.Row{{int64(10)}}, }, { WriteQuery: "INSERT INTO mytable (s, i) VALUES ('x', '64.6');", ExpectedWriteResult: []sql.Row{{sql.NewOkResult(1)}}, SelectQuery: "SELECT i FROM mytable WHERE s = 'x';", ExpectedSelect: []sql.Row{{int64(64)}}, }, { WriteQuery: "INSERT INTO mytable (s, i) VALUES ('x', 999);", ExpectedWriteResult: []sql.Row{{sql.NewOkResult(1)}}, SelectQuery: "SELECT i FROM mytable WHERE s = 'x';", ExpectedSelect: []sql.Row{{int64(999)}}, }, { WriteQuery: "INSERT INTO niltable (i, f) VALUES (10, 10.0), (12, 12.0);", ExpectedWriteResult: []sql.Row{{sql.NewOkResult(2)}}, SelectQuery: "SELECT i,f FROM niltable WHERE f IN (10.0, 12.0) ORDER BY f;", ExpectedSelect: []sql.Row{{int64(10), 10.0}, {int64(12), 12.0}}, }, { WriteQuery: "INSERT INTO mytable SET s = 'x', i = 999;", ExpectedWriteResult: []sql.Row{{sql.NewOkResult(1)}}, SelectQuery: "SELECT i FROM mytable WHERE s = 'x';", ExpectedSelect: []sql.Row{{int64(999)}}, }, { WriteQuery: "INSERT INTO mytable VALUES (999, 'x');", ExpectedWriteResult: []sql.Row{{sql.NewOkResult(1)}}, SelectQuery: "SELECT i FROM mytable WHERE s = 'x';", ExpectedSelect: []sql.Row{{int64(999)}}, }, { WriteQuery: "INSERT INTO mytable SET i = 999, s = 'x';", ExpectedWriteResult: []sql.Row{{sql.NewOkResult(1)}}, SelectQuery: "SELECT i FROM mytable WHERE s = 'x';", ExpectedSelect: []sql.Row{{int64(999)}}, }, { WriteQuery: "INSERT INTO mytable VALUES (999, _binary 'x');", ExpectedWriteResult: []sql.Row{{sql.NewOkResult(1)}}, SelectQuery: "SELECT s FROM mytable WHERE i = 999;", ExpectedSelect: []sql.Row{{"x"}}, }, { WriteQuery: "INSERT INTO mytable SET i = 999, s = _binary 'x';", ExpectedWriteResult: []sql.Row{{sql.NewOkResult(1)}}, SelectQuery: "SELECT s FROM mytable WHERE i = 999;", ExpectedSelect: []sql.Row{{"x"}}, }, { WriteQuery: `INSERT INTO typestable VALUES ( 999, 127, 32767, 2147483647, 9223372036854775807, 255, 65535, 4294967295, 18446744073709551615, 3.40282346638528859811704183484516925440e+38, 1.797693134862315708145274237317043567981e+308, '2037-04-05 12:51:36', '2231-11-07', 'random text', true, '{"key":"value"}', 'blobdata' );`, ExpectedWriteResult: []sql.Row{{sql.NewOkResult(1)}}, SelectQuery: "SELECT * FROM typestable WHERE id = 999;", ExpectedSelect: []sql.Row{{ int64(999), int8(math.MaxInt8), int16(math.MaxInt16), int32(math.MaxInt32), int64(math.MaxInt64), uint8(math.MaxUint8), uint16(math.MaxUint16), uint32(math.MaxUint32), uint64(math.MaxUint64), float32(math.MaxFloat32), float64(math.MaxFloat64), sql.MustConvert(sql.Timestamp.Convert("2037-04-05 12:51:36")), sql.MustConvert(sql.Date.Convert("2231-11-07")), "random text", sql.True, sql.MustJSON(`{"key":"value"}`), "blobdata", }}, }, { WriteQuery: `INSERT INTO typestable SET id = 999, i8 = 127, i16 = 32767, i32 = 2147483647, i64 = 9223372036854775807, u8 = 255, u16 = 65535, u32 = 4294967295, u64 = 18446744073709551615, f32 = 3.40282346638528859811704183484516925440e+38, f64 = 1.797693134862315708145274237317043567981e+308, ti = '2037-04-05 12:51:36', da = '2231-11-07', te = 'random text', bo = true, js = '{"key":"value"}', bl = 'blobdata' ;`, ExpectedWriteResult: []sql.Row{{sql.NewOkResult(1)}}, SelectQuery: "SELECT * FROM typestable WHERE id = 999;", ExpectedSelect: []sql.Row{{ int64(999), int8(math.MaxInt8), int16(math.MaxInt16), int32(math.MaxInt32), int64(math.MaxInt64), uint8(math.MaxUint8), uint16(math.MaxUint16), uint32(math.MaxUint32), uint64(math.MaxUint64), float32(math.MaxFloat32), float64(math.MaxFloat64), sql.MustConvert(sql.Timestamp.Convert("2037-04-05 12:51:36")), sql.MustConvert(sql.Date.Convert("2231-11-07")), "random text", sql.True, sql.MustJSON(`{"key":"value"}`), "blobdata", }}, }, { WriteQuery: `INSERT INTO typestable VALUES ( 999, -128, -32768, -2147483648, -9223372036854775808, 0, 0, 0, 0, 1.401298464324817070923729583289916131280e-45, 4.940656458412465441765687928682213723651e-324, '0000-00-00 00:00:00', '0000-00-00', '', false, '""', '' );`, ExpectedWriteResult: []sql.Row{{sql.NewOkResult(1)}}, SelectQuery: "SELECT * FROM typestable WHERE id = 999;", ExpectedSelect: []sql.Row{{ int64(999), int8(-math.MaxInt8 - 1), int16(-math.MaxInt16 - 1), int32(-math.MaxInt32 - 1), int64(-math.MaxInt64 - 1), uint8(0), uint16(0), uint32(0), uint64(0), float32(math.SmallestNonzeroFloat32), float64(math.SmallestNonzeroFloat64), sql.Timestamp.Zero(), sql.Date.Zero(), "", sql.False, sql.MustJSON(`""`), "", }}, }, { WriteQuery: `INSERT INTO typestable SET id = 999, i8 = -128, i16 = -32768, i32 = -2147483648, i64 = -9223372036854775808, u8 = 0, u16 = 0, u32 = 0, u64 = 0, f32 = 1.401298464324817070923729583289916131280e-45, f64 = 4.940656458412465441765687928682213723651e-324, ti = '0000-00-00 00:00:00', da = '0000-00-00', te = '', bo = false, js = '""', bl = '' ;`, ExpectedWriteResult: []sql.Row{{sql.NewOkResult(1)}}, SelectQuery: "SELECT * FROM typestable WHERE id = 999;", ExpectedSelect: []sql.Row{{ int64(999), int8(-math.MaxInt8 - 1), int16(-math.MaxInt16 - 1), int32(-math.MaxInt32 - 1), int64(-math.MaxInt64 - 1), uint8(0), uint16(0), uint32(0), uint64(0), float32(math.SmallestNonzeroFloat32), float64(math.SmallestNonzeroFloat64), sql.Timestamp.Zero(), sql.Date.Zero(), "", sql.False, sql.MustJSON(`""`), "", }}, }, { WriteQuery: `INSERT INTO typestable SET id = 999, i8 = -128, i16 = -32768, i32 = -2147483648, i64 = -9223372036854775808, u8 = 0, u16 = 0, u32 = 0, u64 = 0, f32 = 1.401298464324817070923729583289916131280e-45, f64 = 4.940656458412465441765687928682213723651e-324, ti = '2037-04-05 12:51:36 -0000 UTC', da = '0000-00-00', te = '', bo = false, js = '""', bl = '' ;`, ExpectedWriteResult: []sql.Row{{sql.NewOkResult(1)}}, SelectQuery: "SELECT * FROM typestable WHERE id = 999;", ExpectedSelect: []sql.Row{{ int64(999), int8(-math.MaxInt8 - 1), int16(-math.MaxInt16 - 1), int32(-math.MaxInt32 - 1), int64(-math.MaxInt64 - 1), uint8(0), uint16(0), uint32(0), uint64(0), float32(math.SmallestNonzeroFloat32), float64(math.SmallestNonzeroFloat64), sql.MustConvert(sql.Timestamp.Convert("2037-04-05 12:51:36")), sql.Date.Zero(), "", sql.False, sql.MustJSON(`""`), "", }}, }, { WriteQuery: `INSERT INTO mytable (i,s) VALUES (10, 'NULL')`, ExpectedWriteResult: []sql.Row{{sql.NewOkResult(1)}}, SelectQuery: "SELECT * FROM mytable WHERE i = 10;", ExpectedSelect: []sql.Row{{int64(10), "NULL"}}, }, { WriteQuery: `INSERT INTO typestable VALUES (999, null, null, null, null, null, null, null, null, null, null, null, null, null, null, null, null);`, ExpectedWriteResult: []sql.Row{{sql.NewOkResult(1)}}, SelectQuery: "SELECT * FROM typestable WHERE id = 999;", ExpectedSelect: []sql.Row{{int64(999), nil, nil, nil, nil, nil, nil, nil, nil, nil, nil, nil, nil, nil, nil, nil, nil}}, }, { WriteQuery: `INSERT INTO typestable (id, ti, da) VALUES (999, '2021-09-1', '2021-9-01');`, ExpectedWriteResult: []sql.Row{{sql.NewOkResult(1)}}, SelectQuery: "SELECT id, ti, da FROM typestable WHERE id = 999;", ExpectedSelect: []sql.Row{{int64(999), sql.MustConvert(sql.Timestamp.Convert("2021-09-01")), sql.MustConvert(sql.Date.Convert("2021-09-01"))}}, }, { WriteQuery: `INSERT INTO typestable SET id=999, i8=null, i16=null, i32=null, i64=null, u8=null, u16=null, u32=null, u64=null, f32=null, f64=null, ti=null, da=null, te=null, bo=null, js=null, bl=null;`, ExpectedWriteResult: []sql.Row{{sql.NewOkResult(1)}}, SelectQuery: "SELECT * FROM typestable WHERE id = 999;", ExpectedSelect: []sql.Row{{int64(999), nil, nil, nil, nil, nil, nil, nil, nil, nil, nil, nil, nil, nil, nil, nil, nil}}, }, { WriteQuery: "INSERT INTO mytable SELECT i+100,s FROM mytable", ExpectedWriteResult: []sql.Row{{sql.NewOkResult(3)}}, SelectQuery: "SELECT * FROM mytable ORDER BY i", ExpectedSelect: []sql.Row{ {int64(1), "first row"}, {int64(2), "second row"}, {int64(3), "third row"}, {int64(101), "first row"}, {int64(102), "second row"}, {int64(103), "third row"}, }, }, { WriteQuery: "INSERT INTO emptytable SELECT * FROM mytable", ExpectedWriteResult: []sql.Row{{sql.NewOkResult(3)}}, SelectQuery: "SELECT * FROM emptytable ORDER BY i", ExpectedSelect: []sql.Row{ {int64(1), "first row"}, {int64(2), "second row"}, {int64(3), "third row"}, }, }, { WriteQuery: "INSERT INTO emptytable SELECT * FROM mytable where mytable.i > 2", ExpectedWriteResult: []sql.Row{{sql.NewOkResult(1)}}, SelectQuery: "SELECT * FROM emptytable ORDER BY i", ExpectedSelect: []sql.Row{ {int64(3), "third row"}, }, }, { WriteQuery: "INSERT INTO niltable (i,f) SELECT i+10, NULL FROM mytable where mytable.i > 2", ExpectedWriteResult: []sql.Row{{sql.NewOkResult(1)}}, SelectQuery: "SELECT * FROM niltable where i > 10 ORDER BY i", ExpectedSelect: []sql.Row{ {13, nil, nil, nil}, }, }, { WriteQuery: "INSERT INTO mytable (i,s) SELECT i+10, 'new' FROM mytable", ExpectedWriteResult: []sql.Row{{sql.NewOkResult(3)}}, SelectQuery: "SELECT * FROM mytable ORDER BY i", ExpectedSelect: []sql.Row{ {int64(1), "first row"}, {int64(2), "second row"}, {int64(3), "third row"}, {int64(11), "new"}, {int64(12), "new"}, {int64(13), "new"}, }, }, { WriteQuery: "INSERT INTO mytable SELECT i2+100, s2 FROM othertable", ExpectedWriteResult: []sql.Row{{sql.NewOkResult(3)}}, SelectQuery: "SELECT * FROM mytable ORDER BY i,s", ExpectedSelect: []sql.Row{ {int64(1), "first row"}, {int64(2), "second row"}, {int64(3), "third row"}, {int64(101), "third"}, {int64(102), "second"}, {int64(103), "first"}, }, }, { WriteQuery: "INSERT INTO emptytable (s,i) SELECT * FROM othertable", ExpectedWriteResult: []sql.Row{{sql.NewOkResult(3)}}, SelectQuery: "SELECT * FROM emptytable ORDER BY i,s", ExpectedSelect: []sql.Row{ {int64(1), "third"}, {int64(2), "second"}, {int64(3), "first"}, }, }, { WriteQuery: "INSERT INTO emptytable (s,i) SELECT concat(m.s, o.s2), m.i FROM othertable o JOIN mytable m ON m.i=o.i2", ExpectedWriteResult: []sql.Row{{sql.NewOkResult(3)}}, SelectQuery: "SELECT * FROM emptytable ORDER BY i,s", ExpectedSelect: []sql.Row{ {int64(1), "first rowthird"}, {int64(2), "second rowsecond"}, {int64(3), "third rowfirst"}, }, }, { WriteQuery: `INSERT INTO emptytable (s,i) SELECT s,i from mytable where i = 1 union select s,i from mytable where i = 3`, ExpectedWriteResult: []sql.Row{{sql.NewOkResult(2)}}, SelectQuery: "SELECT * FROM emptytable ORDER BY i,s", ExpectedSelect: []sql.Row{ {int64(1), "first row"}, {int64(3), "third row"}, }, }, { WriteQuery: `INSERT INTO emptytable (s,i) SELECT s,i from mytable where i = 1 union select s,i from mytable where i = 3 union select s,i from mytable where i > 2`, ExpectedWriteResult: []sql.Row{{sql.NewOkResult(2)}}, SelectQuery: "SELECT * FROM emptytable ORDER BY i,s", ExpectedSelect: []sql.Row{ {int64(1), "first row"}, {int64(3), "third row"}, }, }, { WriteQuery: `INSERT INTO emptytable (s,i) SELECT s,i from mytable where i = 1 union all select s,i+1 from mytable where i < 2 union all select s,i+2 from mytable where i in (1)`, ExpectedWriteResult: []sql.Row{{sql.NewOkResult(3)}}, SelectQuery: "SELECT * FROM emptytable ORDER BY i,s", ExpectedSelect: []sql.Row{ {int64(1), "first row"}, {int64(2), "first row"}, {int64(3), "first row"}, }, }, { WriteQuery: "INSERT INTO emptytable (s,i) SELECT distinct s,i from mytable", ExpectedWriteResult: []sql.Row{{sql.NewOkResult(3)}}, SelectQuery: "SELECT * FROM emptytable ORDER BY i,s", ExpectedSelect: []sql.Row{ {int64(1), "first row"}, {int64(2), "second row"}, {int64(3), "third row"}, }, }, { WriteQuery: "INSERT INTO mytable (i,s) SELECT (i + 10.0) / 10.0 + 10 + i, concat(s, ' new') FROM mytable", ExpectedWriteResult: []sql.Row{{sql.NewOkResult(3)}}, SelectQuery: "SELECT * FROM mytable ORDER BY i, s", ExpectedSelect: []sql.Row{ {int64(1), "first row"}, {int64(2), "second row"}, {int64(3), "third row"}, {int64(12), "first row new"}, {int64(13), "second row new"}, {int64(14), "third row new"}, }, }, { WriteQuery: "INSERT INTO mytable (i,s) SELECT CHAR_LENGTH(s), concat('numrows: ', count(*)) from mytable group by 1", ExpectedWriteResult: []sql.Row{{sql.NewOkResult(2)}}, SelectQuery: "SELECT * FROM mytable ORDER BY i, s", ExpectedSelect: []sql.Row{ {1, "first row"}, {2, "second row"}, {3, "third row"}, {9, "numrows: 2"}, {10, "numrows: 1"}, }, }, { WriteQuery: "INSERT INTO mytable (i,s) SELECT CHAR_LENGTH(s), concat('numrows: ', count(*)) from mytable group by 1 HAVING CHAR_LENGTH(s) > 9", ExpectedWriteResult: []sql.Row{{sql.NewOkResult(1)}}, SelectQuery: "SELECT * FROM mytable ORDER BY i, s", ExpectedSelect: []sql.Row{ {1, "first row"}, {2, "second row"}, {3, "third row"}, {10, "numrows: 1"}, }, }, { WriteQuery: "INSERT INTO mytable (i,s) SELECT i * 2, concat(s,s) from mytable order by 1 desc limit 1", ExpectedWriteResult: []sql.Row{{sql.NewOkResult(1)}}, SelectQuery: "SELECT * FROM mytable ORDER BY i, s", ExpectedSelect: []sql.Row{ {1, "first row"}, {2, "second row"}, {3, "third row"}, {6, "third rowthird row"}, }, }, { WriteQuery: "INSERT INTO mytable (i,s) SELECT i + 3, concat(s,s) from mytable order by 1 desc", ExpectedWriteResult: []sql.Row{{sql.NewOkResult(3)}}, SelectQuery: "SELECT * FROM mytable ORDER BY i, s", ExpectedSelect: []sql.Row{ {1, "first row"}, {2, "second row"}, {3, "third row"}, {4, "first rowfirst row"}, {5, "second rowsecond row"}, {6, "third rowthird row"}, }, }, { WriteQuery: `INSERT INTO mytable (i,s) SELECT sub.i + 10, ot.s2 FROM othertable ot INNER JOIN (SELECT i, i2, s2 FROM mytable INNER JOIN othertable ON i = i2) sub ON sub.i = ot.i2 order by 1`, ExpectedWriteResult: []sql.Row{{sql.NewOkResult(3)}}, SelectQuery: "SELECT * FROM mytable where i > 10 ORDER BY i, s", ExpectedSelect: []sql.Row{ {11, "third"}, {12, "second"}, {13, "first"}, }, }, { WriteQuery: `INSERT INTO mytable (i,s) SELECT sub.i + 10, ot.s2 FROM (SELECT i, i2, s2 FROM mytable INNER JOIN othertable ON i = i2) sub INNER JOIN othertable ot ON sub.i = ot.i2 order by 1`, ExpectedWriteResult: []sql.Row{{sql.NewOkResult(3)}}, SelectQuery: "SELECT * FROM mytable where i > 10 ORDER BY i, s", ExpectedSelect: []sql.Row{ {11, "third"}, {12, "second"}, {13, "first"}, }, }, { WriteQuery: "INSERT INTO mytable (i,s) values (1, 'hello') ON DUPLICATE KEY UPDATE s='hello'", ExpectedWriteResult: []sql.Row{{sql.NewOkResult(2)}}, SelectQuery: "SELECT * FROM mytable WHERE i = 1", ExpectedSelect: []sql.Row{{int64(1), "hello"}}, }, { WriteQuery: "INSERT INTO mytable (i,s) values (1, 'hello2') ON DUPLICATE KEY UPDATE s='hello3'", ExpectedWriteResult: []sql.Row{{sql.NewOkResult(2)}}, SelectQuery: "SELECT * FROM mytable WHERE i = 1", ExpectedSelect: []sql.Row{{int64(1), "hello3"}}, }, { WriteQuery: "INSERT INTO mytable (i,s) values (1, 'hello') ON DUPLICATE KEY UPDATE i=10", ExpectedWriteResult: []sql.Row{{sql.NewOkResult(2)}}, SelectQuery: "SELECT * FROM mytable WHERE i = 10", ExpectedSelect: []sql.Row{{int64(10), "first row"}}, }, { WriteQuery: "INSERT INTO mytable (i,s) values (1, 'hello2') ON DUPLICATE KEY UPDATE s='hello3'", ExpectedWriteResult: []sql.Row{{sql.NewOkResult(2)}}, SelectQuery: "SELECT * FROM mytable WHERE i = 1", ExpectedSelect: []sql.Row{{int64(1), "hello3"}}, }, { WriteQuery: "INSERT INTO mytable (i,s) values (1, 'hello2'), (2, 'hello3'), (4, 'no conflict') ON DUPLICATE KEY UPDATE s='hello4'", ExpectedWriteResult: []sql.Row{{sql.NewOkResult(5)}}, SelectQuery: "SELECT * FROM mytable ORDER BY 1", ExpectedSelect: []sql.Row{ {1, "hello4"}, {2, "hello4"}, {3, "third row"}, {4, "no conflict"}, }, }, { WriteQuery: "INSERT INTO mytable (i,s) values (10, 'hello') ON DUPLICATE KEY UPDATE s='hello'", ExpectedWriteResult: []sql.Row{{sql.NewOkResult(1)}}, SelectQuery: "SELECT * FROM mytable ORDER BY 1", ExpectedSelect: []sql.Row{ {1, "first row"}, {2, "second row"}, {3, "third row"}, {10, "hello"}, }, }, { WriteQuery: "INSERT INTO mytable (i,s) values (1,'hi') ON DUPLICATE KEY UPDATE s=VALUES(s)", ExpectedWriteResult: []sql.Row{{sql.NewOkResult(2)}}, SelectQuery: "SELECT * FROM mytable WHERE i = 1", ExpectedSelect: []sql.Row{{int64(1), "hi"}}, }, { WriteQuery: "INSERT INTO mytable (s,i) values ('dup',1) ON DUPLICATE KEY UPDATE s=CONCAT(VALUES(s), 'licate')", ExpectedWriteResult: []sql.Row{{sql.NewOkResult(2)}}, SelectQuery: "SELECT * FROM mytable WHERE i = 1", ExpectedSelect: []sql.Row{{int64(1), "duplicate"}}, }, { WriteQuery: "INSERT INTO mytable (i,s) values (1,'mar'), (2,'par') ON DUPLICATE KEY UPDATE s=CONCAT(VALUES(s), 'tial')", ExpectedWriteResult: []sql.Row{{sql.NewOkResult(4)}}, SelectQuery: "SELECT * FROM mytable WHERE i IN (1,2) ORDER BY i", ExpectedSelect: []sql.Row{{int64(1), "martial"}, {int64(2), "partial"}}, }, { WriteQuery: "INSERT INTO mytable (i,s) values (1,'maybe') ON DUPLICATE KEY UPDATE i=VALUES(i)+8000, s=VALUES(s)", ExpectedWriteResult: []sql.Row{{sql.NewOkResult(2)}}, SelectQuery: "SELECT * FROM mytable WHERE i = 8001", ExpectedSelect: []sql.Row{{int64(8001), "maybe"}}, }, { WriteQuery: "INSERT INTO auto_increment_tbl (c0) values (44)", ExpectedWriteResult: []sql.Row{{sql.OkResult{RowsAffected: 1, InsertID: 4}}}, SelectQuery: "SELECT * FROM auto_increment_tbl ORDER BY pk", ExpectedSelect: []sql.Row{ {1, 11}, {2, 22}, {3, 33}, {4, 44}, }, }, { WriteQuery: "INSERT INTO auto_increment_tbl (c0) values (44),(55)", ExpectedWriteResult: []sql.Row{{sql.OkResult{RowsAffected: 2, InsertID: 4}}}, SelectQuery: "SELECT * FROM auto_increment_tbl ORDER BY pk", ExpectedSelect: []sql.Row{ {1, 11}, {2, 22}, {3, 33}, {4, 44}, {5, 55}, }, }, { WriteQuery: "INSERT INTO auto_increment_tbl values (NULL, 44)", ExpectedWriteResult: []sql.Row{{sql.OkResult{RowsAffected: 1, InsertID: 4}}}, SelectQuery: "SELECT * FROM auto_increment_tbl ORDER BY pk", ExpectedSelect: []sql.Row{ {1, 11}, {2, 22}, {3, 33}, {4, 44}, }, }, { WriteQuery: "INSERT INTO auto_increment_tbl values (0, 44)", ExpectedWriteResult: []sql.Row{{sql.OkResult{RowsAffected: 1, InsertID: 4}}}, SelectQuery: "SELECT * FROM auto_increment_tbl ORDER BY pk", ExpectedSelect: []sql.Row{ {1, 11}, {2, 22}, {3, 33}, {4, 44}, }, }, { WriteQuery: "INSERT INTO auto_increment_tbl values (5, 44)", ExpectedWriteResult: []sql.Row{{sql.OkResult{RowsAffected: 1, InsertID: 5}}}, SelectQuery: "SELECT * FROM auto_increment_tbl ORDER BY pk", ExpectedSelect: []sql.Row{ {1, 11}, {2, 22}, {3, 33}, {5, 44}, }, }, { WriteQuery: "INSERT INTO auto_increment_tbl values " + "(NULL, 44), (NULL, 55), (9, 99), (NULL, 110), (NULL, 121)", ExpectedWriteResult: []sql.Row{{sql.OkResult{RowsAffected: 5, InsertID: 4}}}, SelectQuery: "SELECT * FROM auto_increment_tbl ORDER BY pk", ExpectedSelect: []sql.Row{ {1, 11}, {2, 22}, {3, 33}, {4, 44}, {5, 55}, {9, 99}, {10, 110}, {11, 121}, }, }, { WriteQuery: `INSERT INTO auto_increment_tbl (c0) SELECT 44 FROM dual`, ExpectedWriteResult: []sql.Row{{sql.OkResult{RowsAffected: 1, InsertID: 4}}}, SelectQuery: "SELECT * FROM auto_increment_tbl", ExpectedSelect: []sql.Row{ {1, 11}, {2, 22}, {3, 33}, {4, 44}, }, }, { WriteQuery: `INSERT INTO othertable VALUES ("fourth", 1) ON DUPLICATE KEY UPDATE s2="fourth"`, ExpectedWriteResult: []sql.Row{{sql.NewOkResult(2)}}, SelectQuery: "SELECT * FROM othertable", ExpectedSelect: []sql.Row{ sql.NewRow("first", int64(3)), sql.NewRow("second", int64(2)), sql.NewRow("fourth", int64(1)), }, }, { WriteQuery: `INSERT INTO othertable(S2,I2) values ('fourth',0)`, ExpectedWriteResult: []sql.Row{{sql.NewOkResult(1)}}, SelectQuery: `SELECT * FROM othertable where s2='fourth'`, ExpectedSelect: []sql.Row{ {"fourth", 0}, }, }, { WriteQuery: `INSERT INTO auto_increment_tbl VALUES ('4', 44)`, ExpectedWriteResult: []sql.Row{ {sql.OkResult{InsertID: 4, RowsAffected: 1}}, }, SelectQuery: `SELECT * from auto_increment_tbl where pk=4`, ExpectedSelect: []sql.Row{ {4, 44}, }, }, { WriteQuery: `INSERT INTO keyless (c0, c1) SELECT * from keyless where c0=0 and c1=0`, ExpectedWriteResult: []sql.Row{{sql.NewOkResult(1)}}, SelectQuery: `SELECT * from keyless where c0=0`, ExpectedSelect: []sql.Row{ {0, 0}, {0, 0}, }, }, { WriteQuery: `insert into keyless (c0, c1) select a.c0, a.c1 from (select 1, 1) as a(c0, c1) join keyless on a.c0 = keyless.c0`, ExpectedWriteResult: []sql.Row{{sql.NewOkResult(2)}}, SelectQuery: `SELECT * from keyless where c0=1`, ExpectedSelect: []sql.Row{ {1, 1}, {1, 1}, {1, 1}, {1, 1}, }, }, }
var InsertScripts = []ScriptTest{ { Name: "insert into sparse auto_increment table", SetUpScript: []string{ "create table auto (pk int primary key auto_increment)", "insert into auto values (10), (20), (30)", "insert into auto values (NULL)", "insert into auto values (40)", "insert into auto values (0)", }, Assertions: []ScriptTestAssertion{ { Query: "select * from auto order by 1", Expected: []sql.Row{ {10}, {20}, {30}, {31}, {40}, {41}, }, }, }, }, { Name: "insert negative values into auto_increment values", SetUpScript: []string{ "create table auto (pk int primary key auto_increment)", "insert into auto values (10), (20), (30)", "insert into auto values (-1), (-2), (-3)", "insert into auto () values ()", "insert into auto values (0), (0), (0)", }, Assertions: []ScriptTestAssertion{ { Query: "select * from auto order by 1", Expected: []sql.Row{ {-3}, {-2}, {-1}, {10}, {20}, {30}, {31}, {32}, {33}, {34}, }, }, }, }, { Name: "insert into auto_increment unique key column", SetUpScript: []string{ "create table auto (pk int primary key, npk int unique auto_increment)", "insert into auto (pk) values (10), (20), (30)", }, Assertions: []ScriptTestAssertion{ { Query: "select * from auto order by 1", Expected: []sql.Row{ {10, 1}, {20, 2}, {30, 3}, }, }, }, }, { Name: "insert into auto_increment with multiple unique key columns", SetUpScript: []string{ "create table auto (pk int primary key, npk1 int auto_increment, npk2 int, unique(npk1, npk2))", "insert into auto (pk) values (10), (20), (30)", }, Assertions: []ScriptTestAssertion{ { Query: "select * from auto order by 1", Expected: []sql.Row{ {10, 1, nil}, {20, 2, nil}, {30, 3, nil}, }, }, }, }, { Name: "insert into auto_increment key/index column", SetUpScript: []string{ "create table auto_no_primary (i int auto_increment, index(i))", "insert into auto_no_primary (i) values (0), (0), (0)", }, Assertions: []ScriptTestAssertion{ { Query: "select * from auto_no_primary order by 1", Expected: []sql.Row{ {1}, {2}, {3}, }, }, }, }, { Name: "insert into auto_increment with multiple key/index columns", SetUpScript: []string{ "create table auto_no_primary (i int auto_increment, j int, index(i))", "insert into auto_no_primary (i) values (0), (0), (0)", }, Assertions: []ScriptTestAssertion{ { Query: "select * from auto_no_primary order by 1", Expected: []sql.Row{ {1, nil}, {2, nil}, {3, nil}, }, }, }, }, { Name: "auto increment table handles deletes", SetUpScript: []string{ "create table auto (pk int primary key auto_increment)", "insert into auto values (10)", "delete from auto where pk = 10", "insert into auto values (NULL)", }, Assertions: []ScriptTestAssertion{ { Query: "select * from auto order by 1", Expected: []sql.Row{ {11}, }, }, }, }, { Name: "create auto_increment table with out-of-line primary key def", SetUpScript: []string{ `create table auto ( pk int auto_increment, c0 int, primary key(pk) );`, "insert into auto values (NULL,10), (NULL,20), (NULL,30)", }, Assertions: []ScriptTestAssertion{ { Query: "select * from auto order by 1", Expected: []sql.Row{ {1, 10}, {2, 20}, {3, 30}, }, }, }, }, { Name: "alter auto_increment value", SetUpScript: []string{ `create table auto ( pk int auto_increment, c0 int, primary key(pk) );`, "insert into auto values (NULL,10), (NULL,20), (NULL,30)", "alter table auto auto_increment 9;", "insert into auto values (NULL,90)", }, Assertions: []ScriptTestAssertion{ { Query: "select * from auto order by 1", Expected: []sql.Row{ {1, 10}, {2, 20}, {3, 30}, {9, 90}, }, }, }, }, { Name: "alter auto_increment value to float", SetUpScript: []string{ `create table auto ( pk int auto_increment, c0 int, primary key(pk) );`, "insert into auto values (NULL,10), (NULL,20), (NULL,30)", "alter table auto auto_increment = 19.9;", "insert into auto values (NULL,190)", }, Assertions: []ScriptTestAssertion{ { Query: "select * from auto order by 1", Expected: []sql.Row{ {1, 10}, {2, 20}, {3, 30}, {19, 190}, }, }, }, }, { Name: "auto increment on tinyint", SetUpScript: []string{ "create table auto (pk tinyint primary key auto_increment)", "insert into auto values (NULL),(10),(0)", }, Assertions: []ScriptTestAssertion{ { Query: "select * from auto order by 1", Expected: []sql.Row{ {1}, {10}, {11}, }, }, }, }, { Name: "auto increment on smallint", SetUpScript: []string{ "create table auto (pk smallint primary key auto_increment)", "insert into auto values (NULL),(10),(0)", }, Assertions: []ScriptTestAssertion{ { Query: "select * from auto order by 1", Expected: []sql.Row{ {1}, {10}, {11}, }, }, }, }, { Name: "auto increment on mediumint", SetUpScript: []string{ "create table auto (pk mediumint primary key auto_increment)", "insert into auto values (NULL),(10),(0)", }, Assertions: []ScriptTestAssertion{ { Query: "select * from auto order by 1", Expected: []sql.Row{ {1}, {10}, {11}, }, }, }, }, { Name: "auto increment on int", SetUpScript: []string{ "create table auto (pk int primary key auto_increment)", "insert into auto values (NULL),(10),(0)", }, Assertions: []ScriptTestAssertion{ { Query: "select * from auto order by 1", Expected: []sql.Row{ {1}, {10}, {11}, }, }, }, }, { Name: "auto increment on bigint", SetUpScript: []string{ "create table auto (pk bigint primary key auto_increment)", "insert into auto values (NULL),(10),(0)", }, Assertions: []ScriptTestAssertion{ { Query: "select * from auto order by 1", Expected: []sql.Row{ {1}, {10}, {11}, }, }, }, }, { Name: "auto increment on tinyint unsigned", SetUpScript: []string{ "create table auto (pk tinyint unsigned primary key auto_increment)", "insert into auto values (NULL),(10),(0)", }, Assertions: []ScriptTestAssertion{ { Query: "select * from auto order by 1", Expected: []sql.Row{ {uint64(1)}, {uint64(10)}, {uint64(11)}, }, }, }, }, { Name: "auto increment on smallint unsigned", SetUpScript: []string{ "create table auto (pk smallint unsigned primary key auto_increment)", "insert into auto values (NULL),(10),(0)", }, Assertions: []ScriptTestAssertion{ { Query: "select * from auto order by 1", Expected: []sql.Row{ {uint64(1)}, {uint64(10)}, {uint64(11)}, }, }, }, }, { Name: "auto increment on mediumint unsigned", SetUpScript: []string{ "create table auto (pk mediumint unsigned primary key auto_increment)", "insert into auto values (NULL),(10),(0)", }, Assertions: []ScriptTestAssertion{ { Query: "select * from auto order by 1", Expected: []sql.Row{ {uint64(1)}, {uint64(10)}, {uint64(11)}, }, }, }, }, { Name: "auto increment on int unsigned", SetUpScript: []string{ "create table auto (pk int unsigned primary key auto_increment)", "insert into auto values (NULL),(10),(0)", }, Assertions: []ScriptTestAssertion{ { Query: "select * from auto order by 1", Expected: []sql.Row{ {uint64(1)}, {uint64(10)}, {uint64(11)}, }, }, }, }, { Name: "auto increment on bigint unsigned", SetUpScript: []string{ "create table auto (pk bigint unsigned primary key auto_increment)", "insert into auto values (NULL),(10),(0)", }, Assertions: []ScriptTestAssertion{ { Query: "select * from auto order by 1", Expected: []sql.Row{ {uint64(1)}, {uint64(10)}, {uint64(11)}, }, }, }, }, { Name: "auto increment on float", SetUpScript: []string{ "create table auto (pk float primary key auto_increment)", "insert into auto values (NULL),(10),(0)", }, Assertions: []ScriptTestAssertion{ { Query: "select * from auto order by 1", Expected: []sql.Row{ {float64(1)}, {float64(10)}, {float64(11)}, }, }, }, }, { Name: "auto increment on double", SetUpScript: []string{ "create table auto (pk double primary key auto_increment)", "insert into auto values (NULL),(10),(0)", }, Assertions: []ScriptTestAssertion{ { Query: "select * from auto order by 1", Expected: []sql.Row{ {float64(1)}, {float64(10)}, {float64(11)}, }, }, }, }, { Name: "explicit DEFAULT", SetUpScript: []string{ "CREATE TABLE mytable(id int PRIMARY KEY, v2 int NOT NULL DEFAULT '2')", }, Assertions: []ScriptTestAssertion{ { Query: "INSERT INTO mytable (id, v2)values (1, DEFAULT)", Expected: []sql.Row{ {sql.OkResult{RowsAffected: 1}}, }, }, { Query: "SELECT * FROM mytable", Expected: []sql.Row{ {1, 2}, }, }, }, }, { Name: "explicit DEFAULT with multiple values", SetUpScript: []string{ "CREATE TABLE mytable(id int PRIMARY KEY, v2 int NOT NULL DEFAULT '2')", }, Assertions: []ScriptTestAssertion{ { Query: "INSERT INTO mytable (id, v2)values (1, DEFAULT), (2, DEFAULT)", Expected: []sql.Row{ {sql.OkResult{RowsAffected: 2}}, }, }, { Query: "SELECT * FROM mytable", Expected: []sql.Row{ {1, 2}, {2, 2}, }, }, }, }, { Name: "Try INSERT IGNORE with primary key, non null, and single row violations", SetUpScript: []string{ "CREATE TABLE y (pk int primary key, c1 int NOT NULL);", "INSERT IGNORE INTO y VALUES (1, 1), (1,2), (2, 2), (3, 3)", }, Assertions: []ScriptTestAssertion{ { Query: "SELECT * FROM y", Expected: []sql.Row{ {1, 1}, {2, 2}, {3, 3}, }, }, { Query: "INSERT IGNORE INTO y VALUES (1, 2), (4,4)", Expected: []sql.Row{ {sql.OkResult{RowsAffected: 1}}, }, ExpectedWarning: mysql.ERDupEntry, }, { Query: "INSERT IGNORE INTO y VALUES (5, NULL)", Expected: []sql.Row{ {sql.OkResult{RowsAffected: 1}}, }, ExpectedWarning: mysql.ERBadNullError, }, { Query: "INSERT IGNORE INTO y SELECT * FROM y WHERE pk=(SELECT pk FROM y WHERE pk > 1);", Expected: []sql.Row{ {sql.OkResult{RowsAffected: 0}}, }, ExpectedWarning: mysql.ERSubqueryNo1Row, }, { Query: "INSERT IGNORE INTO y SELECT 10, 0 FROM dual WHERE 1=(SELECT 1 FROM dual UNION SELECT 2 FROM dual);", Expected: []sql.Row{ {sql.OkResult{RowsAffected: 0}}, }, ExpectedWarning: mysql.ERSubqueryNo1Row, }, { Query: "INSERT IGNORE INTO y SELECT 11, 0 FROM dual WHERE 1=(SELECT 1 FROM dual UNION SELECT 2 FROM dual) UNION SELECT 12, 0 FROM dual;", Expected: []sql.Row{ {sql.OkResult{RowsAffected: 1}}, }, ExpectedWarning: mysql.ERSubqueryNo1Row, }, { Query: "INSERT IGNORE INTO y SELECT 13, 0 FROM dual UNION SELECT 14, 0 FROM dual WHERE 1=(SELECT 1 FROM dual UNION SELECT 2 FROM dual);", Expected: []sql.Row{ {sql.OkResult{RowsAffected: 1}}, }, ExpectedWarning: mysql.ERSubqueryNo1Row, }, { Query: "INSERT IGNORE INTO y VALUES (3, 8)", Expected: []sql.Row{ {sql.OkResult{RowsAffected: 0}}, }, ExpectedWarning: mysql.ERDupEntry, }, }, }, { Name: "INSERT Accumulator tests", SetUpScript: []string{ "CREATE TABLE test(pk int primary key, val int)", "INSERT INTO test values (1,1)", }, Assertions: []ScriptTestAssertion{ { Query: `INSERT INTO test VALUES (2,2),(2,3)`, ExpectedErr: sql.ErrPrimaryKeyViolation, }, { Query: `DELETE FROM test where pk = 1;`, Expected: []sql.Row{{sql.OkResult{RowsAffected: 1}}}, }, { Query: `INSERT INTO test VALUES (1,1)`, Expected: []sql.Row{ {sql.OkResult{RowsAffected: 1}}, }, }, }, }, { Name: "INSERT Case Sensitivity", SetUpScript: []string{ "CREATE TABLE test (PK int PRIMARY KEY);", }, Assertions: []ScriptTestAssertion{ { Query: "insert into test(pk) values (1)", Expected: []sql.Row{{sql.NewOkResult(1)}}, }, }, }, { Name: "INSERT string with exact char length but extra byte length", SetUpScript: []string{ "CREATE TABLE city (id int PRIMARY KEY, district char(20) NOT NULL DEFAULT '');", }, Assertions: []ScriptTestAssertion{ { Query: "INSERT INTO city VALUES (1,'San Pedro de Macorís');", Expected: []sql.Row{{sql.NewOkResult(1)}}, }, }, }, { Name: "Insert on duplicate key", SetUpScript: []string{ `CREATE TABLE users ( id varchar(42) PRIMARY KEY )`, `CREATE TABLE nodes ( id varchar(42) PRIMARY KEY, owner varchar(42), status varchar(12), timestamp bigint NOT NULL, FOREIGN KEY(owner) REFERENCES users(id) )`, "INSERT INTO users values ('milo'), ('dabe')", "INSERT INTO nodes values ('id1', 'milo', 'off', 1)", }, Assertions: []ScriptTestAssertion{ { Query: "insert into nodes(id,owner,status,timestamp) values('id1','dabe','off',2) on duplicate key update owner='milo',status='on'", Expected: []sql.Row{ {sql.OkResult{RowsAffected: 2}}, }, }, { Query: "insert into nodes(id,owner,status,timestamp) values('id2','dabe','off',3) on duplicate key update owner='milo',status='on'", Expected: []sql.Row{ {sql.OkResult{RowsAffected: 1}}, }, }, { Query: "select * from nodes", Expected: []sql.Row{ {"id1", "milo", "on", 1}, {"id2", "dabe", "off", 3}, }, }, }, }, }
var JoinQueryTests = []QueryTest{ { Query: "select a.pk, c.v2 from one_pk_three_idx a cross join one_pk_three_idx b left join one_pk_three_idx c on b.pk = c.v2 where b.pk = 0 and a.v2 = 1;", Expected: []sql.Row{ {2, 0}, {2, 0}, {2, 0}, {2, 0}, }, }, { Query: "select a.pk, c.v2 from one_pk_three_idx a cross join one_pk_three_idx b right join one_pk_three_idx c on b.pk = c.v3 where b.pk = 0 and c.v2 = 0 order by a.pk;", Expected: []sql.Row{ {0, 0}, {0, 0}, {1, 0}, {1, 0}, {2, 0}, {2, 0}, {3, 0}, {3, 0}, {4, 0}, {4, 0}, {5, 0}, {5, 0}, {6, 0}, {6, 0}, {7, 0}, {7, 0}, }, }, { Query: "select a.pk, c.v2 from one_pk_three_idx a cross join one_pk_three_idx b inner join (select * from one_pk_three_idx where v2 = 0) c on b.pk = c.v3 where b.pk = 0 and c.v2 = 0 order by a.pk;", Expected: []sql.Row{ {0, 0}, {0, 0}, {1, 0}, {1, 0}, {2, 0}, {2, 0}, {3, 0}, {3, 0}, {4, 0}, {4, 0}, {5, 0}, {5, 0}, {6, 0}, {6, 0}, {7, 0}, {7, 0}, }, }, { Query: "select a.pk, c.v2 from one_pk_three_idx a cross join one_pk_three_idx b left join one_pk_three_idx c on b.pk = c.v1+1 where b.pk = 0 order by a.pk;", Expected: []sql.Row{ {0, nil}, {1, nil}, {2, nil}, {3, nil}, {4, nil}, {5, nil}, {6, nil}, {7, nil}, }, }, { Query: "select a.pk, c.v2 from one_pk_three_idx a cross join one_pk_three_idx b right join one_pk_three_idx c on b.pk = c.v1 where b.pk = 0 and c.v2 = 0 order by a.pk;", Expected: []sql.Row{ {0, 0}, {0, 0}, {1, 0}, {1, 0}, {2, 0}, {2, 0}, {3, 0}, {3, 0}, {4, 0}, {4, 0}, {5, 0}, {5, 0}, {6, 0}, {6, 0}, {7, 0}, {7, 0}, }, }, { Query: "select * from mytable a CROSS JOIN mytable b RIGHT JOIN mytable c ON b.i = c.i + 1;", Expected: []sql.Row{ {1, "first row", 2, "second row", 1, "first row"}, {2, "second row", 2, "second row", 1, "first row"}, {3, "third row", 2, "second row", 1, "first row"}, {1, "first row", 3, "third row", 2, "second row"}, {2, "second row", 3, "third row", 2, "second row"}, {3, "third row", 3, "third row", 2, "second row"}, {nil, nil, nil, nil, 3, "third row"}, }, }, { Query: "select * from mytable a CROSS JOIN mytable b LEFT JOIN mytable c ON b.i = c.i + 1;", Expected: []sql.Row{ {3, "third row", 1, "first row", nil, nil}, {2, "second row", 1, "first row", nil, nil}, {1, "first row", 1, "first row", nil, nil}, {3, "third row", 2, "second row", 1, "first row"}, {2, "second row", 2, "second row", 1, "first row"}, {1, "first row", 2, "second row", 1, "first row"}, {3, "third row", 3, "third row", 2, "second row"}, {2, "second row", 3, "third row", 2, "second row"}, {1, "first row", 3, "third row", 2, "second row"}, }, }, { Query: "select * from mytable a CROSS JOIN mytable b LEFT JOIN mytable c ON b.i+1 = c.i;", Expected: []sql.Row{ {3, "third row", 1, "first row", 2, "second row"}, {2, "second row", 1, "first row", 2, "second row"}, {1, "first row", 1, "first row", 2, "second row"}, {3, "third row", 2, "second row", 3, "third row"}, {2, "second row", 2, "second row", 3, "third row"}, {1, "first row", 2, "second row", 3, "third row"}, {3, "third row", 3, "third row", nil, nil}, {2, "second row", 3, "third row", nil, nil}, {1, "first row", 3, "third row", nil, nil}, }}, { Query: "select * from mytable a LEFT JOIN mytable b on a.i = b.i LEFT JOIN mytable c ON b.i = c.i + 1;", Expected: []sql.Row{ {1, "first row", 1, "first row", nil, nil}, {2, "second row", 2, "second row", 1, "first row"}, {3, "third row", 3, "third row", 2, "second row"}, }, }, { Query: "select * from mytable a LEFT JOIN mytable b on a.i = b.i RIGHT JOIN mytable c ON b.i = c.i + 1;", Expected: []sql.Row{ {2, "second row", 2, "second row", 1, "first row"}, {3, "third row", 3, "third row", 2, "second row"}, {nil, nil, nil, nil, 3, "third row"}, }, }, { Query: "select * from mytable a RIGHT JOIN mytable b on a.i = b.i RIGHT JOIN mytable c ON b.i = c.i + 1;", Expected: []sql.Row{ {2, "second row", 2, "second row", 1, "first row"}, {3, "third row", 3, "third row", 2, "second row"}, {nil, nil, nil, nil, 3, "third row"}, }, }, { Query: "select * from mytable a RIGHT JOIN mytable b on a.i = b.i LEFT JOIN mytable c ON b.i = c.i + 1;", Expected: []sql.Row{ {1, "first row", 1, "first row", nil, nil}, {2, "second row", 2, "second row", 1, "first row"}, {3, "third row", 3, "third row", 2, "second row"}, }, }, { Query: "select * from mytable a LEFT JOIN mytable b on a.i = b.i LEFT JOIN mytable c ON b.i+1 = c.i;", Expected: []sql.Row{ {1, "first row", 1, "first row", 2, "second row"}, {2, "second row", 2, "second row", 3, "third row"}, {3, "third row", 3, "third row", nil, nil}, }}, { Query: "select * from mytable a LEFT JOIN mytable b on a.i = b.i RIGHT JOIN mytable c ON b.i+1 = c.i;", Expected: []sql.Row{ {nil, nil, nil, nil, 1, "first row"}, {1, "first row", 1, "first row", 2, "second row"}, {2, "second row", 2, "second row", 3, "third row"}, }}, { Query: "select * from mytable a RIGHT JOIN mytable b on a.i = b.i RIGHT JOIN mytable c ON b.i+1= c.i;", Expected: []sql.Row{ {nil, nil, nil, nil, 1, "first row"}, {1, "first row", 1, "first row", 2, "second row"}, {2, "second row", 2, "second row", 3, "third row"}, }}, { Query: "select * from mytable a RIGHT JOIN mytable b on a.i = b.i LEFT JOIN mytable c ON b.i+1 = c.i;", Expected: []sql.Row{ {1, "first row", 1, "first row", 2, "second row"}, {2, "second row", 2, "second row", 3, "third row"}, {3, "third row", 3, "third row", nil, nil}, }, }, { Query: "select * from mytable a CROSS JOIN mytable b RIGHT JOIN mytable c ON b.i+1 = c.i;", Expected: []sql.Row{ {nil, nil, nil, nil, 1, "first row"}, {1, "first row", 1, "first row", 2, "second row"}, {2, "second row", 1, "first row", 2, "second row"}, {3, "third row", 1, "first row", 2, "second row"}, {1, "first row", 2, "second row", 3, "third row"}, {2, "second row", 2, "second row", 3, "third row"}, {3, "third row", 2, "second row", 3, "third row"}, }, }, { Query: "with a as (select a.i, a.s from mytable a CROSS JOIN mytable b) select * from a RIGHT JOIN mytable c on a.i+1 = c.i-1;", Expected: []sql.Row{ {nil, nil, 1, "first row"}, {nil, nil, 2, "second row"}, {1, "first row", 3, "third row"}, {1, "first row", 3, "third row"}, {1, "first row", 3, "third row"}, }, }, { Query: "select a.* from mytable a RIGHT JOIN mytable b on a.i = b.i+1 LEFT JOIN mytable c on a.i = c.i-1 RIGHT JOIN mytable d on b.i = d.i;", Expected: []sql.Row{ {2, "second row"}, {3, "third row"}, {nil, nil}, }, }, { Query: "select a.*,b.* from mytable a RIGHT JOIN othertable b on a.i = b.i2+1 LEFT JOIN mytable c on a.i = c.i-1 LEFT JOIN othertable d on b.i2 = d.i2;", Expected: []sql.Row{ {2, "second row", "third", 1}, {3, "third row", "second", 2}, {nil, nil, "first", 3}, }, }, { Query: "select a.*,b.* from mytable a RIGHT JOIN othertable b on a.i = b.i2+1 RIGHT JOIN mytable c on a.i = c.i-1 LEFT JOIN othertable d on b.i2 = d.i2;", Expected: []sql.Row{ {nil, nil, nil, nil}, {nil, nil, nil, nil}, {2, "second row", "third", 1}, }, }, { Query: "select i.pk, j.v3 from one_pk_two_idx i JOIN one_pk_three_idx j on i.v1 = j.pk;", Expected: []sql.Row{{0, 0}, {1, 1}, {2, 0}, {3, 2}, {4, 0}, {5, 3}, {6, 0}, {7, 4}}, }, { Query: "select i.pk, j.v3, k.c1 from one_pk_two_idx i JOIN one_pk_three_idx j on i.v1 = j.pk JOIN one_pk k on j.v3 = k.pk;", Expected: []sql.Row{{0, 0, 0}, {1, 1, 10}, {2, 0, 0}, {3, 2, 20}, {4, 0, 0}, {5, 3, 30}, {6, 0, 0}}, }, { Query: "select i.pk, j.v3 from (one_pk_two_idx i JOIN one_pk_three_idx j on((i.v1 = j.pk)));", Expected: []sql.Row{{0, 0}, {1, 1}, {2, 0}, {3, 2}, {4, 0}, {5, 3}, {6, 0}, {7, 4}}, }, { Query: "select i.pk, j.v3, k.c1 from ((one_pk_two_idx i JOIN one_pk_three_idx j on ((i.v1 = j.pk))) JOIN one_pk k on((j.v3 = k.pk)));", Expected: []sql.Row{{0, 0, 0}, {1, 1, 10}, {2, 0, 0}, {3, 2, 20}, {4, 0, 0}, {5, 3, 30}, {6, 0, 0}}, }, { Query: "select i.pk, j.v3, k.c1 from (one_pk_two_idx i JOIN one_pk_three_idx j on ((i.v1 = j.pk)) JOIN one_pk k on((j.v3 = k.pk)));", Expected: []sql.Row{{0, 0, 0}, {1, 1, 10}, {2, 0, 0}, {3, 2, 20}, {4, 0, 0}, {5, 3, 30}, {6, 0, 0}}, }, { Query: "select a.* from one_pk_two_idx a RIGHT JOIN (one_pk_two_idx i JOIN one_pk_three_idx j on i.v1 = j.pk) on a.pk = i.v1 LEFT JOIN (one_pk_two_idx k JOIN one_pk_three_idx l on k.v1 = l.pk) on a.pk = l.v2;", Expected: []sql.Row{{0, 0, 0}, {0, 0, 0}, {0, 0, 0}, {0, 0, 0}, {1, 1, 1}, {2, 2, 2}, {3, 3, 3}, {4, 4, 4}, {5, 5, 5}, {6, 6, 6}, {7, 7, 7}}, }, { Query: "select a.* from one_pk_two_idx a LEFT JOIN (one_pk_two_idx i JOIN one_pk_three_idx j on i.pk = j.v3) on a.pk = i.pk RIGHT JOIN (one_pk_two_idx k JOIN one_pk_three_idx l on k.v2 = l.v3) on a.v1 = l.v2;", Expected: []sql.Row{{0, 0, 0}, {0, 0, 0}, {0, 0, 0}, {0, 0, 0}, {0, 0, 0}, {0, 0, 0}, {0, 0, 0}, {0, 0, 0}, {1, 1, 1}, {2, 2, 2}, {0, 0, 0}, {0, 0, 0}, {0, 0, 0}, {0, 0, 0}, {0, 0, 0}, {0, 0, 0}, {0, 0, 0}, {0, 0, 0}, {3, 3, 3}, {4, 4, 4}, }, }, }
var JsonScripts = []ScriptTest{ { Name: "JSON_ARRAYAGG on one column", SetUpScript: []string{ "create table t (o_id int primary key)", "INSERT INTO t VALUES (1),(2)", }, Assertions: []ScriptTestAssertion{ { Query: "SELECT JSON_ARRAYAGG(o_id) FROM (SELECT * FROM t ORDER BY o_id) as sub", Expected: []sql.Row{ { sql.MustJSON(`[1,2]`), }, }, }, }, }, { Name: "Simple JSON_ARRAYAGG on two columns", SetUpScript: []string{ "create table t (o_id int primary key, attribute longtext)", "INSERT INTO t VALUES (1, 'color'), (2, 'fabric')", }, Assertions: []ScriptTestAssertion{ { Query: "SELECT JSON_ARRAYAGG(o_id), JSON_ARRAYAGG(`attribute`) FROM (SELECT * FROM t ORDER BY o_id) as sub;", Expected: []sql.Row{ { sql.MustJSON(`[1,2]`), sql.MustJSON(`["color","fabric"]`), }, }, }, }, }, { Name: "JSON_ARRAYAGG on column with string values w/ groupby", SetUpScript: []string{ "create table t (o_id int primary key, c0 int, attribute longtext, value longtext)", "INSERT INTO t VALUES (1, 2, 'color', 'red'), (2, 2, 'fabric', 'silk')", }, Assertions: []ScriptTestAssertion{ { Query: "SELECT c0, JSON_ARRAYAGG(`attribute`) FROM (SELECT * FROM t ORDER BY o_id) as sub GROUP BY c0", Expected: []sql.Row{ { 2, sql.MustJSON(`["color","fabric"]`), }, }, }, { Query: "SELECT c0, JSON_ARRAYAGG(value) FROM (SELECT * FROM t ORDER BY o_id) as sub GROUP BY c0", Expected: []sql.Row{ { 2, sql.MustJSON(`["red","silk"]`), }, }, }, }, }, { Name: "JSON_ARRAYAGG on column with int values w/ groupby", SetUpScript: []string{ "create table t2 (o_id int primary key, val int)", "INSERT INTO t2 VALUES (1,1), (2,1), (3,1)", }, Assertions: []ScriptTestAssertion{ { Query: "SELECT val, JSON_ARRAYAGG(o_id) FROM (SELECT * FROM t2 ORDER BY o_id) AS sub GROUP BY val", Expected: []sql.Row{ { 1, sql.MustJSON(`[1,2,3]`), }, }, }, }, }, { Name: "JSON_ARRAYAGG on unknown column throws error", SetUpScript: []string{ "create table t2 (o_id int primary key, val int)", "INSERT INTO t2 VALUES (1,1), (2,2), (3,3)", }, Assertions: []ScriptTestAssertion{ { Query: "SELECT o_id, JSON_ARRAYAGG(val2) FROM t2 GROUP BY o_id", ExpectedErr: sql.ErrColumnNotFound, }, }, }, { Name: "JSON_ARRAYAGG on column with no rows returns NULL", SetUpScript: []string{ "create table t2 (o_id int primary key)", }, Assertions: []ScriptTestAssertion{ { Query: "SELECT JSON_ARRAYAGG(o_id) FROM t2", Expected: []sql.Row{ { sql.MustJSON(`[]`), }, }, }, }, }, { Name: "JSON_ARRAYAGG on row with 1 value, 1 null is fine", SetUpScript: []string{ "create table x(pk int primary key, c1 int)", "INSERT INTO x VALUES (1,NULL)", }, Assertions: []ScriptTestAssertion{ { Query: "SELECT pk, JSON_ARRAYAGG(c1) FROM x GROUP BY pk", Expected: []sql.Row{ {1, sql.MustJSON(`[null]`)}, }, }, { Query: "SELECT JSON_ARRAYAGG(c1) FROM x", Expected: []sql.Row{ {sql.MustJSON(`[null]`)}, }, }, }, }, { Name: "JSON_ARRAYAGG and group by use the same field.", SetUpScript: []string{ "create table x(pk int primary key, c1 int)", "INSERT INTO x VALUES (1, 1)", "INSERT INTO x VALUES (2, 1)", "INSERT INTO x VALUES (3, 3)", "INSERT INTO x VALUES (4, 3)", "INSERT INTO x VALUES (5, 5)", }, Assertions: []ScriptTestAssertion{ { Query: "SELECT JSON_ARRAYAGG(pk) FROM (SELECT * FROM x ORDER BY pk) as sub GROUP BY c1", Expected: []sql.Row{ {sql.MustJSON(`[1,2]`)}, {sql.MustJSON(`[3,4]`)}, {sql.MustJSON(`[5]`)}, }, }, }, }, { Name: "JSON_ARRAGG with simple and nested json objects.", SetUpScript: []string{ "create table j(pk int primary key, field JSON)", `INSERT INTO j VALUES(1, '{"key1": {"key": "value"}}')`, `INSERT INTO j VALUES(2, '{"key1": "value1", "key2": "value2"}')`, `INSERT INTO j VALUES(3, '{"key1": {"key": [2,3]}}')`, `INSERT INTO j VALUES(4, '["a", 1]')`, }, Assertions: []ScriptTestAssertion{ { Query: "SELECT pk, JSON_ARRAYAGG(field) FROM (SELECT * FROM j ORDER BY pk) as sub GROUP BY field ORDER BY pk", Expected: []sql.Row{ {1, sql.MustJSON(`[{"key1": {"key": "value"}}]`)}, {2, sql.MustJSON(`[{"key1": "value1", "key2": "value2"}]`)}, {3, sql.MustJSON(`[{"key1":{"key":[2,3]}}]`)}, {4, sql.MustJSON(`[["a",1]]`)}, }, }, }, }, { Name: "Simple JSON_OBJECTAGG with GROUP BY", SetUpScript: []string{ "create table t2 (o_id int primary key, val int)", "INSERT INTO t2 VALUES (1,1), (2,1), (3,1)", }, Assertions: []ScriptTestAssertion{ { Query: "SELECT JSON_OBJECTAGG(val, o_id) FROM (SELECT * FROM t2 ORDER BY o_id) as sub GROUP BY val", Expected: []sql.Row{ {sql.MustJSON(`{"1": 3}`)}, }, }, }, }, { Name: "More complex JSON_OBJECTAGG WITH GROUP BY", SetUpScript: []string{ "create table t (o_id int primary key, c0 int, attribute longtext, value longtext)", "INSERT INTO t VALUES (1, 2, 'color', 'red'), (2, 2, 'fabric', 'silk')", "INSERT INTO t VALUES (3, 3, 'color', 'green'), (4, 3, 'shape', 'square')", }, Assertions: []ScriptTestAssertion{ { Query: "SELECT c0, JSON_OBJECTAGG(`attribute`, value) FROM (SELECT * FROM t ORDER BY o_id) as sub GROUP BY c0", Expected: []sql.Row{ {2, sql.MustJSON(`{"color": "red", "fabric": "silk"}`)}, {3, sql.MustJSON(`{"color": "green", "shape": "square"}`)}, }, }, { Query: `SELECT c0, JSON_OBJECTAGG(c0, value) FROM (SELECT * FROM t ORDER BY o_id) as sub GROUP BY c0`, Expected: []sql.Row{ {2, sql.MustJSON(`{"2": "silk"}`)}, {3, sql.MustJSON(`{"3": "square"}`)}, }, }, }, }, { Name: "3 column table that uses JSON_OBJECTAGG without groupby", SetUpScript: []string{ "create table t (o_id int primary key, c0 int, attribute longtext, value longtext)", "INSERT INTO t VALUES (1, 2, 'color', 'red'), (2, 2, 'fabric', 'silk')", "INSERT INTO t VALUES (3, 3, 'color', 'green'), (4, 3, 'shape', 'square')", }, Assertions: []ScriptTestAssertion{ { Query: `select JSON_OBJECTAGG(c0, value) from (SELECT * FROM t ORDER BY o_id) as sub`, Expected: []sql.Row{ {sql.MustJSON(`{"2": "silk", "3": "square"}`)}, }, }, { Query: "select JSON_OBJECTAGG(`attribute`, value) from (SELECT * FROM t ORDER BY o_id) as sub", Expected: []sql.Row{ {sql.MustJSON(`{"color": "green", "fabric": "silk", "shape": "square"}`)}, }, }, }, }, { Name: "JSON_OBJECTAGG and null values", SetUpScript: []string{ `create table test (pk int primary key, val longtext)`, `insert into test values (1, NULL)`, }, Assertions: []ScriptTestAssertion{ { Query: `SELECT JSON_OBJECTAGG(pk, val) from test`, Expected: []sql.Row{ {sql.MustJSON(`{"1": null}`)}, }, }, }, }, { Name: "JSON_OBJECTAGG and nested json values", SetUpScript: []string{ "create table j(pk int primary key, c0 int, val JSON)", `INSERT INTO j VALUES(1, 1, '{"key1": "value1", "key2": "value2"}')`, `INSERT INTO j VALUES(2, 1, '{"key1": {"key": [2,3]}}')`, `INSERT INTO j VALUES(3, 2, '["a", 1]')`, }, Assertions: []ScriptTestAssertion{ { Query: `SELECT JSON_OBJECTAGG(c0, val) from (SELECT * FROM j ORDER BY pk) as sub`, Expected: []sql.Row{ {sql.MustJSON(`{"1": {"key1": {"key": [2, 3]}}, "2": ["a", 1]}`)}, }, }, }, }, { Name: "JSON_OBJECTAGG correctly returns null when no rows are present", SetUpScript: []string{ `create table test (pk int primary key, val longtext)`, }, Assertions: []ScriptTestAssertion{ { Query: `SELECT JSON_OBJECTAGG(pk, val) from test`, Expected: []sql.Row{ {nil}, }, }, }, }, { Name: "JSON_OBJECTAGG handles errors appropriately", SetUpScript: []string{ `create table test (pk int primary key, c0 int, val longtext)`, `insert into test values (1, 1, NULL)`, `insert into test values (2, NULL, 1)`, }, Assertions: []ScriptTestAssertion{ { Query: `SELECT JSON_OBJECTAGG(c0, notval) from test`, ExpectedErr: sql.ErrColumnNotFound, }, { Query: `SELECT JSON_OBJECTAGG(notpk, val) from test`, ExpectedErr: sql.ErrColumnNotFound, }, { Query: `SELECT JSON_OBJECTAGG(c0, val) from nottest`, ExpectedErr: sql.ErrTableNotFound, }, { Query: `SELECT JSON_OBJECTAGG(c0, val, badarg) from test`, ExpectedErr: sql.ErrInvalidArgumentNumber, }, { Query: `SELECT JSON_OBJECTAGG(c0) from test`, ExpectedErr: sql.ErrInvalidArgumentNumber, }, { Query: `SELECT JSON_OBJECTAGG(c0, val) from test`, ExpectedErr: sql.ErrJSONObjectAggNullKey, }, }, }, }
var KeylessQueries = []QueryTest{ { Query: "SELECT * FROM keyless ORDER BY c0", Expected: []sql.Row{ {0, 0}, {1, 1}, {1, 1}, {2, 2}, }, }, { Query: "SELECT * FROM keyless ORDER BY c1 DESC", Expected: []sql.Row{ {2, 2}, {1, 1}, {1, 1}, {0, 0}, }, }, { Query: "SELECT * FROM keyless JOIN myTable where c0 = i", Expected: []sql.Row{ {1, 1, 1, "first row"}, {1, 1, 1, "first row"}, {2, 2, 2, "second row"}, }, }, { Query: "SELECT * FROM myTable JOIN keyless WHERE i = c0 ORDER BY i", Expected: []sql.Row{ {1, "first row", 1, 1}, {1, "first row", 1, 1}, {2, "second row", 2, 2}, }, }, { Query: "DESCRIBE keyless", Expected: []sql.Row{ {"c0", "bigint", "YES", "", "", ""}, {"c1", "bigint", "YES", "", "", ""}, }, }, { Query: "SHOW COLUMNS FROM keyless", Expected: []sql.Row{ {"c0", "bigint", "YES", "", "", ""}, {"c1", "bigint", "YES", "", "", ""}, }, }, { Query: "SHOW FULL COLUMNS FROM keyless", Expected: []sql.Row{ {"c0", "bigint", nil, "YES", "", "", "", "", ""}, {"c1", "bigint", nil, "YES", "", "", "", "", ""}, }, }, { Query: "SHOW CREATE TABLE keyless", Expected: []sql.Row{ {"keyless", "CREATE TABLE `keyless` (\n `c0` bigint,\n `c1` bigint\n) ENGINE=InnoDB DEFAULT CHARSET=utf8mb4 COLLATE=utf8mb4_0900_bin"}, }, }, }
var LoadDataErrorScripts = []ScriptTest{ { Name: "Load data into table that doesn't exist throws error.", Query: "LOAD DATA INFILE 'test1.txt' INTO TABLE loadtable", ExpectedErr: sql.ErrTableNotFound, }, { Name: "Load data with unknown files throws an error.", SetUpScript: []string{ "create table loadtable(pk longtext primary key, c1 int)", }, Assertions: []ScriptTestAssertion{ { Query: "LOAD DATA INFILE '/x/ytx' INTO TABLE loadtable", ExpectedErr: sql.ErrLoadDataCannotOpen, }, }, }, { Name: "Load data with unknown columns throws an error", SetUpScript: []string{ "create table loadtable(pk int primary key)", }, Assertions: []ScriptTestAssertion{ { Query: "LOAD DATA INFILE './testdata/test1.txt' INTO TABLE loadtable FIELDS ENCLOSED BY '\"' (bad)", ExpectedErr: plan.ErrInsertIntoNonexistentColumn, }, }, }, { Name: "Load data escaped by terms longer than 1 character throws an error", SetUpScript: []string{ "create table loadtable(pk int primary key)", }, Assertions: []ScriptTestAssertion{ { Query: "LOAD DATA INFILE './testdata/test1.txt' INTO TABLE loadtable FIELDS ESCAPED BY 'xx' (pk)", ExpectedErr: sql.ErrLoadDataCharacterLength, }, }, }, { Name: "Load data enclosed by term longer than 1 character throws an error", SetUpScript: []string{ "create table loadtable(pk int primary key)", }, Assertions: []ScriptTestAssertion{ { Query: "LOAD DATA INFILE './testdata/test1.txt' INTO TABLE loadtable FIELDS ENCLOSED BY 'xx' (pk)", ExpectedErr: sql.ErrLoadDataCharacterLength, }, }, }, }
var LoadDataFailingScripts = []ScriptTest{ { Name: "Escaped values are correctly parsed.", SetUpScript: []string{ "create table loadtable(pk longtext)", "LOAD DATA INFILE 'test5.txt' INTO TABLE loadtable FIELDS ENCLOSED BY '\"' IGNORE 1 LINES", }, Assertions: []ScriptTestAssertion{ { Query: "select * from loadtable", Expected: []sql.Row{{"hi"}, {"hello"}, {nil}, {"TryN"}, {fmt.Sprintf("%c", 26)}, {fmt.Sprintf("%c", 0)}, {"new\n"}}, }, }, }, { Name: "Load and terminate have the same values.", SetUpScript: []string{ "create table loadtable(pk int primary key)", "LOAD DATA INFILE 'test1.txt' INTO TABLE loadtable FIELDS TERMINATED BY '\"' ENCLOSED BY '\"'", }, Assertions: []ScriptTestAssertion{ { Query: "select * from loadtable", Expected: []sql.Row{{int8(1)}, {int8(2)}, {int8(3)}, {int8(4)}}, }, }, }, { Name: "Loading value into different column type results in default value.", SetUpScript: []string{ "create table loadtable(pk longtext, c1 int)", "LOAD DATA INFILE 'test4.txt' INTO TABLE loadtable FIELDS ENCLOSED BY '\"' (c1)", }, Assertions: []ScriptTestAssertion{ { Query: "select * from loadtable", Expected: []sql.Row{{nil, 0}, {nil, 0}}, }, }, }, { Name: "LOAD DATA handles nulls", SetUpScript: []string{ "create table loadtable(pk longtext, c1 int)", "LOAD DATA INFILE 'test4.txt' INTO TABLE loadtable FIELDS ENCLOSED BY '\"'", }, Assertions: []ScriptTestAssertion{ { Query: "select * from loadtable", Expected: []sql.Row{{"hi", 1}, {"hello", nil}}, }, }, }, { Name: "LOAD DATA can handle a differing column order", SetUpScript: []string{ "create table loadtable(pk int, c1 string) ", "LOAD DATA INFILE 'test4.txt' INTO TABLE loadtable FIELDS ENCLOSED BY '\"' (c1, pk)", }, Assertions: []ScriptTestAssertion{ { Query: "select * from loadtable", Expected: []sql.Row{{1, "hi"}, {nil, "hello"}}, }, }, }, }
var LoadDataScripts = []ScriptTest{ { Name: "Basic load data with enclosed values.", SetUpScript: []string{ "create table loadtable(pk int primary key)", "LOAD DATA INFILE './testdata/test1.txt' INTO TABLE loadtable FIELDS ENCLOSED BY '\"'", }, Assertions: []ScriptTestAssertion{ { Query: "select * from loadtable", Expected: []sql.Row{{int8(1)}, {int8(2)}, {int8(3)}, {int8(4)}}, }, }, }, { Name: "Load data with csv", SetUpScript: []string{ "create table loadtable(pk int primary key, c1 longtext)", "LOAD DATA INFILE './testdata/test2.csv' INTO TABLE loadtable FIELDS TERMINATED BY ',' IGNORE 1 LINES", }, Assertions: []ScriptTestAssertion{ { Query: "select * from loadtable", Expected: []sql.Row{{int8(1), "hi"}, {int8(2), "hello"}}, }, }, }, { Name: "Load data with csv with prefix.", SetUpScript: []string{ "create table loadtable(pk longtext primary key, c1 int)", "LOAD DATA INFILE './testdata/test3.csv' INTO TABLE loadtable FIELDS TERMINATED BY ',' LINES STARTING BY 'xxx' IGNORE 1 LINES (`pk`, `c1`)", }, Assertions: []ScriptTestAssertion{ { Query: "select * from loadtable", Expected: []sql.Row{{"\"abc\"", int8(1)}, {"\"def\"", int8(2)}, {"\"hello\"", nil}}, }, }, }, { Name: "LOAD DATA with all columns reordered in projection", SetUpScript: []string{ "create table loadtable(pk longtext primary key, c1 int)", "LOAD DATA INFILE './testdata/test3backwards.csv' INTO TABLE loadtable FIELDS TERMINATED BY ',' LINES STARTING BY 'xxx' IGNORE 1 LINES (`c1`, `pk`)", }, Assertions: []ScriptTestAssertion{ { Query: "select * from loadtable", Expected: []sql.Row{{"\"abc\"", int8(1)}, {"\"def\"", int8(2)}, {"\"hello\"", nil}}, }, }, }, { Name: "Table has more columns than import.", SetUpScript: []string{ "create table loadtable(pk int primary key, c1 int)", "LOAD DATA INFILE './testdata/test1.txt' INTO TABLE loadtable FIELDS ENCLOSED BY '\"'", }, Assertions: []ScriptTestAssertion{ { Query: "select * from loadtable ORDER BY pk", Expected: []sql.Row{{1, nil}, {2, nil}, {3, nil}, {4, nil}}, }, }, }, { Name: "LOAD DATA handles Windows line-endings and a subset of columns that are not in order", SetUpScript: []string{ "CREATE TABLE inmate_population_snapshots (id char(21) NOT NULL, snapshot_date date NOT NULL, total int," + "total_off_site int, male int, female int, other_gender int, white int, black int, hispanic int," + "asian int, american_indian int, mexican_american int, multi_racial int, other_race int," + "on_probation int, on_parole int, felony int, misdemeanor int, other_offense int," + "convicted_or_sentenced int, detained_or_awaiting_trial int, first_time_incarcerated int, employed int," + "unemployed int, citizen int, noncitizen int, juvenile int, juvenile_male int, juvenile_female int," + "death_row_condemned int, solitary_confinement int, technical_parole_violators int," + "source_url varchar(2043) NOT NULL, source_url_2 varchar(2043), civil_offense int, federal_offense int," + "PRIMARY KEY (id,snapshot_date), KEY id (id));", "LOAD DATA INFILE './testdata/test6.csv' INTO TABLE inmate_population_snapshots " + "FIELDS TERMINATED BY ',' " + "LINES TERMINATED BY '\r\n' " + "IGNORE 1 LINES " + "(federal_offense, misdemeanor, total, detained_or_awaiting_trial, felony, snapshot_date, id, source_url, source_url_2)", }, Assertions: []ScriptTestAssertion{ { Query: "SELECT * FROM inmate_population_snapshots", Expected: []sql.Row{ {"8946", time.Date(2020, 5, 1, 0, 0, 0, 0, time.UTC), nil, nil, nil, nil, nil, nil, nil, nil, nil, nil, nil, nil, nil, nil, nil, 0, nil, nil, nil, 0, nil, nil, nil, nil, nil, nil, nil, nil, nil, nil, nil, "https://www.website.gov", "https://www.website.gov/other.html", nil, nil}, {"8976", time.Date(2020, 5, 1, 0, 0, 0, 0, time.UTC), 196, nil, nil, nil, nil, nil, nil, nil, nil, nil, nil, nil, nil, nil, nil, 0, 73, nil, nil, 123, nil, nil, nil, nil, nil, nil, nil, nil, nil, nil, nil, "https://www.website.gov", "https://www.website.gov/other.html", nil, 0}, {"8978", time.Date(2020, 5, 1, 0, 0, 0, 0, time.UTC), 0, nil, nil, nil, nil, nil, nil, nil, nil, nil, nil, nil, nil, nil, nil, 0, nil, nil, nil, 0, nil, nil, nil, nil, nil, nil, nil, nil, nil, nil, nil, "https://www.website.gov", "https://www.website.gov/other.html", nil, nil}, {"8979", time.Date(2020, 5, 1, 0, 0, 0, 0, time.UTC), 71, nil, nil, nil, nil, nil, nil, nil, nil, nil, nil, nil, nil, nil, nil, 5, 3, nil, nil, 63, nil, nil, nil, nil, nil, nil, nil, nil, nil, nil, nil, "https://www.website.gov", "https://www.website.gov/other.html", nil, 0}, }, }, }, }, { Name: "LOAD DATA handles non-nil default values", SetUpScript: []string{ "CREATE TABLE test1 (pk BIGINT PRIMARY KEY, v1 BIGINT DEFAULT (v2 * 10), v2 BIGINT DEFAULT 5);", "CREATE TABLE test2 (pk BIGINT PRIMARY KEY, v1 BIGINT DEFAULT (v2 * 10), v2 BIGINT DEFAULT 5);", "CREATE TABLE test3 (pk BIGINT PRIMARY KEY, v1 BIGINT DEFAULT (pk * 10), v2 BIGINT DEFAULT (v1 - 1));", "CREATE TABLE test4 (pk BIGINT PRIMARY KEY, v1 BIGINT DEFAULT (pk * 10), v2 BIGINT DEFAULT (v1 - 1));", "LOAD DATA INFILE './testdata/test7.txt' INTO TABLE test1;", "LOAD DATA INFILE './testdata/test7.txt' INTO TABLE test2 (pk);", "LOAD DATA INFILE './testdata/test7.txt' INTO TABLE test3;", "LOAD DATA INFILE './testdata/test7.txt' INTO TABLE test4 (pk);", }, Assertions: []ScriptTestAssertion{ { Query: "SELECT * FROM test1", Expected: []sql.Row{ {1, 50, 5}, {2, 50, 5}, {3, 50, 5}, }, }, { Query: "SELECT * FROM test2", Expected: []sql.Row{ {1, 50, 5}, {2, 50, 5}, {3, 50, 5}, }, }, { Query: "SELECT * FROM test3", Expected: []sql.Row{ {1, 10, 9}, {2, 20, 19}, {3, 30, 29}, }, }, { Query: "SELECT * FROM test4", Expected: []sql.Row{ {1, 10, 9}, {2, 20, 19}, {3, 30, 29}, }, }, }, }, { Name: "LOAD DATA handles non-nil default values with varying field counts per row", SetUpScript: []string{ "CREATE TABLE test1 (pk BIGINT PRIMARY KEY, v1 BIGINT DEFAULT (v2 * 10), v2 BIGINT DEFAULT 5);", "CREATE TABLE test2 (pk BIGINT PRIMARY KEY, v1 BIGINT DEFAULT (pk * 10), v2 BIGINT DEFAULT (v1 - 1));", "LOAD DATA INFILE './testdata/test8.txt' INTO TABLE test1 FIELDS TERMINATED BY ',';", "LOAD DATA INFILE './testdata/test8.txt' INTO TABLE test2 FIELDS TERMINATED BY ',';", }, Assertions: []ScriptTestAssertion{ { Query: "SELECT * FROM test1", Expected: []sql.Row{ {1, 50, 5}, {2, 100, 5}, {3, 50, 5}, }, }, { Query: "SELECT * FROM test2", Expected: []sql.Row{ {1, 10, 9}, {2, 100, 99}, {3, 30, 29}, }, }, }, }, }
var NullRangeTests = []QueryTest{ { Query: "select * from null_ranges where y IS NULL or y < 1", Expected: []sql.Row{ {0, 0}, {3, nil}, {4, nil}, }, }, { Query: "select * from null_ranges where y IS NULL and y < 1", Expected: []sql.Row{}, }, { Query: "select * from null_ranges where y IS NULL or y IS NOT NULL", Expected: []sql.Row{ {0, 0}, {1, 1}, {2, 2}, {3, nil}, {4, nil}, }, }, { Query: "select * from null_ranges where y IS NOT NULL", Expected: []sql.Row{ {0, 0}, {1, 1}, {2, 2}, }, }, { Query: "select * from null_ranges where y IS NULL or y = 0 or y = 1", Expected: []sql.Row{ {0, 0}, {1, 1}, {3, nil}, {4, nil}, }, }, { Query: "select * from null_ranges where y IS NULL or y < 1 or y > 1", Expected: []sql.Row{ {0, 0}, {2, 2}, {3, nil}, {4, nil}, }, }, { Query: "select * from null_ranges where y IS NOT NULL and x > 1", Expected: []sql.Row{ {2, 2}, }, }, { Query: "select * from null_ranges where y IS NULL and x = 4", Expected: []sql.Row{ {4, nil}, }, }, { Query: "select * from null_ranges where y IS NULL and x > 1", Expected: []sql.Row{ {3, nil}, {4, nil}, }, }, { Query: "select * from null_ranges where y IS NULL and y IS NOT NULL", Expected: []sql.Row{}, }, { Query: "select * from null_ranges where y is NULL and y > -1 and y > -2", Expected: []sql.Row{}, }, { Query: "select * from null_ranges where y > -1 and y < 7 and y IS NULL", Expected: []sql.Row{}, }, { Query: "select * from null_ranges where y > -1 and y > -2 and y IS NOT NULL", Expected: []sql.Row{ {0, 0}, {1, 1}, {2, 2}, }, }, { Query: "select * from null_ranges where y > -1 and y > 1 and y IS NOT NULL", Expected: []sql.Row{ {2, 2}, }, }, { Query: "select * from null_ranges where y < 6 and y > -1 and y IS NOT NULL", Expected: []sql.Row{ {0, 0}, {1, 1}, {2, 2}, }, }, }
var OrdinalDDLQueries = []QueryTest{ { Query: "show keys from short_ord_pk", Expected: []sql.Row{ {"short_ord_pk", 0, "PRIMARY", 1, "y", nil, 0, nil, nil, "", "BTREE", "", "", "YES", nil}, {"short_ord_pk", 0, "PRIMARY", 2, "x", nil, 0, nil, nil, "", "BTREE", "", "", "YES", nil}, }, }, { Query: "SELECT column_name, ordinal_position FROM INFORMATION_SCHEMA.COLUMNS WHERE TABLE_SCHEMA = 'mydb' AND TABLE_NAME = 'short_ord_pk'", Expected: []sql.Row{ {"x", uint(1)}, {"y", uint(2)}, }, }, { Query: "show keys from long_ord_pk1", Expected: []sql.Row{ {"long_ord_pk1", 0, "PRIMARY", 1, "y", nil, 0, nil, nil, "", "BTREE", "", "", "YES", nil}, {"long_ord_pk1", 0, "PRIMARY", 2, "v", nil, 0, nil, nil, "", "BTREE", "", "", "YES", nil}, }, }, { Query: "SELECT column_name, ordinal_position FROM INFORMATION_SCHEMA.COLUMNS WHERE TABLE_SCHEMA = 'mydb' AND TABLE_NAME = 'long_ord_pk1' and column_key = 'PRI'", Expected: []sql.Row{ {"v", uint(2)}, {"y", uint(5)}, }, }, { Query: "show keys from long_ord_pk2", Expected: []sql.Row{ {"long_ord_pk2", 0, "PRIMARY", 1, "y", nil, 0, nil, nil, "", "BTREE", "", "", "YES", nil}, {"long_ord_pk2", 0, "PRIMARY", 2, "v", nil, 0, nil, nil, "", "BTREE", "", "", "YES", nil}, {"long_ord_pk2", 0, "PRIMARY", 3, "x", nil, 0, nil, nil, "", "BTREE", "", "", "YES", nil}, {"long_ord_pk2", 0, "PRIMARY", 4, "z", nil, 0, nil, nil, "", "BTREE", "", "", "YES", nil}, {"long_ord_pk2", 0, "PRIMARY", 5, "u", nil, 0, nil, nil, "", "BTREE", "", "", "YES", nil}, }, }, { Query: "SELECT column_name, ordinal_position FROM INFORMATION_SCHEMA.COLUMNS WHERE TABLE_SCHEMA = 'mydb' AND TABLE_NAME = 'long_ord_pk2' and column_key = 'PRI'", Expected: []sql.Row{ {"u", uint(1)}, {"v", uint(2)}, {"x", uint(4)}, {"y", uint(5)}, {"z", uint(6)}, }, }, { Query: "show keys from long_ord_pk3", Expected: []sql.Row{ {"long_ord_pk3", 0, "PRIMARY", 1, "y", nil, 0, nil, nil, "", "BTREE", "", "", "YES", nil}, {"long_ord_pk3", 0, "PRIMARY", 2, "v", nil, 0, nil, nil, "", "BTREE", "", "", "YES", nil}, {"long_ord_pk3", 0, "PRIMARY", 3, "x", nil, 0, nil, nil, "", "BTREE", "", "", "YES", nil}, {"long_ord_pk3", 0, "PRIMARY", 4, "z", nil, 0, nil, nil, "", "BTREE", "", "", "YES", nil}, {"long_ord_pk3", 0, "PRIMARY", 5, "u", nil, 0, nil, nil, "", "BTREE", "", "", "YES", nil}, }, }, { Query: "SELECT column_name, ordinal_position FROM INFORMATION_SCHEMA.COLUMNS WHERE TABLE_SCHEMA = 'mydb' AND TABLE_NAME = 'long_ord_pk3' and column_key = 'PRI'", Expected: []sql.Row{ {"u", uint(1)}, {"v", uint(2)}, {"x", uint(5)}, {"y", uint(6)}, {"z", uint(7)}, }, }, { Query: "show keys from ord_kl", Expected: []sql.Row{}, }, { Query: "SELECT column_name, ordinal_position FROM INFORMATION_SCHEMA.COLUMNS WHERE TABLE_SCHEMA = 'mydb' AND TABLE_NAME = 'ord_kl' and column_key = 'PRI'", Expected: []sql.Row{}, }, }
var OrdinalDDLWriteQueries = []WriteQueryTest{ { WriteQuery: "ALTER TABLE long_ord_pk1 ADD COLUMN ww int AFTER v", ExpectedWriteResult: []sql.Row{ {sql.OkResult{RowsAffected: 0}}, }, SelectQuery: "SELECT column_name, ordinal_position FROM INFORMATION_SCHEMA.COLUMNS WHERE TABLE_SCHEMA = 'mydb' AND TABLE_NAME = 'long_ord_pk1' and column_key = 'PRI'", ExpectedSelect: []sql.Row{ {"v", uint(2)}, {"y", uint(6)}, }, }, { WriteQuery: "ALTER TABLE long_ord_pk1 MODIFY COLUMN w int AFTER y", ExpectedWriteResult: []sql.Row{ {sql.OkResult{RowsAffected: 0}}, }, SelectQuery: "SELECT column_name, ordinal_position FROM INFORMATION_SCHEMA.COLUMNS WHERE TABLE_SCHEMA = 'mydb' AND TABLE_NAME = 'long_ord_pk1' and column_key = 'PRI'", ExpectedSelect: []sql.Row{ {"v", uint(2)}, {"y", uint(4)}, }, }, { WriteQuery: "ALTER TABLE long_ord_pk1 DROP PRIMARY KEY", ExpectedWriteResult: []sql.Row{ {sql.OkResult{RowsAffected: 0}}, }, SelectQuery: "show keys from ord_kl", ExpectedSelect: []sql.Row{}, }, { WriteQuery: "ALTER TABLE ord_kl ADD PRIMARY KEY (y,v)", ExpectedWriteResult: []sql.Row{ {sql.OkResult{RowsAffected: 0}}, }, SelectQuery: "show keys from ord_kl", ExpectedSelect: []sql.Row{ {"ord_kl", 0, "PRIMARY", 1, "y", nil, 0, nil, nil, "", "BTREE", "", "", "YES", nil}, {"ord_kl", 0, "PRIMARY", 2, "v", nil, 0, nil, nil, "", "BTREE", "", "", "YES", nil}, }, }, { WriteQuery: "ALTER TABLE ord_kl ADD PRIMARY KEY (y,v)", ExpectedWriteResult: []sql.Row{ {sql.OkResult{RowsAffected: 0}}, }, SelectQuery: "SELECT column_name, ordinal_position FROM INFORMATION_SCHEMA.COLUMNS WHERE TABLE_SCHEMA = 'mydb' AND TABLE_NAME = 'ord_kl' and column_key = 'PRI'", ExpectedSelect: []sql.Row{ {"v", uint(2)}, {"y", uint(5)}, }, }, { WriteQuery: "ALTER TABLE long_ord_pk1 MODIFY COLUMN y int AFTER u", ExpectedWriteResult: []sql.Row{ {sql.OkResult{RowsAffected: 0}}, }, SelectQuery: `SELECT column_name, ordinal_position FROM INFORMATION_SCHEMA.COLUMNS WHERE TABLE_SCHEMA = 'mydb' AND TABLE_NAME = 'long_ord_pk1' and column_key = 'PRI' order by 2`, ExpectedSelect: []sql.Row{ {"y", uint(2)}, {"v", uint(3)}, }, }, { WriteQuery: "ALTER TABLE long_ord_pk1 MODIFY COLUMN y int AFTER u", ExpectedWriteResult: []sql.Row{ {sql.OkResult{RowsAffected: 0}}, }, SelectQuery: "show keys from long_ord_pk1", ExpectedSelect: []sql.Row{ {"long_ord_pk1", 0, "PRIMARY", 1, "y", nil, 0, nil, nil, "", "BTREE", "", "", "YES", nil}, {"long_ord_pk1", 0, "PRIMARY", 2, "v", nil, 0, nil, nil, "", "BTREE", "", "", "YES", nil}, }, }, { WriteQuery: "ALTER TABLE long_ord_pk1 RENAME COLUMN y to yy", ExpectedWriteResult: []sql.Row{ {sql.OkResult{RowsAffected: 0}}, }, SelectQuery: "SELECT column_name, ordinal_position FROM INFORMATION_SCHEMA.COLUMNS WHERE TABLE_SCHEMA = 'mydb' AND TABLE_NAME = 'long_ord_pk1' and column_key = 'PRI'", ExpectedSelect: []sql.Row{ {"v", uint(2)}, {"yy", uint(5)}, }, }, { WriteQuery: "ALTER TABLE long_ord_pk1 RENAME COLUMN y to yy", ExpectedWriteResult: []sql.Row{ {sql.OkResult{RowsAffected: 0}}, }, SelectQuery: "show keys from long_ord_pk1", ExpectedSelect: []sql.Row{ {"long_ord_pk1", 0, "PRIMARY", 1, "yy", nil, 0, nil, nil, "", "BTREE", "", "", "YES", nil}, {"long_ord_pk1", 0, "PRIMARY", 2, "v", nil, 0, nil, nil, "", "BTREE", "", "", "YES", nil}, }, }, { WriteQuery: "ALTER TABLE long_ord_pk2 ADD COLUMN ww int AFTER w", ExpectedWriteResult: []sql.Row{ {sql.OkResult{RowsAffected: 0}}, }, SelectQuery: "SELECT column_name, ordinal_position FROM INFORMATION_SCHEMA.COLUMNS WHERE TABLE_SCHEMA = 'mydb' AND TABLE_NAME = 'long_ord_pk2' and column_key = 'PRI'", ExpectedSelect: []sql.Row{ {"u", uint(1)}, {"v", uint(2)}, {"x", uint(5)}, {"y", uint(6)}, {"z", uint(7)}, }, }, { WriteQuery: "ALTER TABLE long_ord_pk2 ADD COLUMN ww int AFTER w", ExpectedWriteResult: []sql.Row{ {sql.OkResult{RowsAffected: 0}}, }, SelectQuery: "show keys from long_ord_pk2", ExpectedSelect: []sql.Row{ {"long_ord_pk2", 0, "PRIMARY", 1, "y", nil, 0, nil, nil, "", "BTREE", "", "", "YES", nil}, {"long_ord_pk2", 0, "PRIMARY", 2, "v", nil, 0, nil, nil, "", "BTREE", "", "", "YES", nil}, {"long_ord_pk2", 0, "PRIMARY", 3, "x", nil, 0, nil, nil, "", "BTREE", "", "", "YES", nil}, {"long_ord_pk2", 0, "PRIMARY", 4, "z", nil, 0, nil, nil, "", "BTREE", "", "", "YES", nil}, {"long_ord_pk2", 0, "PRIMARY", 5, "u", nil, 0, nil, nil, "", "BTREE", "", "", "YES", nil}, }, }, { WriteQuery: "ALTER TABLE long_ord_pk3 DROP COLUMN ww", ExpectedWriteResult: []sql.Row{ {sql.OkResult{RowsAffected: 0}}, }, SelectQuery: "SELECT column_name, ordinal_position FROM INFORMATION_SCHEMA.COLUMNS WHERE TABLE_SCHEMA = 'mydb' AND TABLE_NAME = 'long_ord_pk3' and column_key = 'PRI'", ExpectedSelect: []sql.Row{ {"u", uint(1)}, {"v", uint(2)}, {"x", uint(4)}, {"y", uint(5)}, {"z", uint(6)}, }, }, { WriteQuery: "ALTER TABLE long_ord_pk3 DROP COLUMN ww", ExpectedWriteResult: []sql.Row{ {sql.OkResult{RowsAffected: 0}}, }, SelectQuery: "show keys from long_ord_pk3", ExpectedSelect: []sql.Row{ {"long_ord_pk3", 0, "PRIMARY", 1, "y", nil, 0, nil, nil, "", "BTREE", "", "", "YES", nil}, {"long_ord_pk3", 0, "PRIMARY", 2, "v", nil, 0, nil, nil, "", "BTREE", "", "", "YES", nil}, {"long_ord_pk3", 0, "PRIMARY", 3, "x", nil, 0, nil, nil, "", "BTREE", "", "", "YES", nil}, {"long_ord_pk3", 0, "PRIMARY", 4, "z", nil, 0, nil, nil, "", "BTREE", "", "", "YES", nil}, {"long_ord_pk3", 0, "PRIMARY", 5, "u", nil, 0, nil, nil, "", "BTREE", "", "", "YES", nil}, }, }, { WriteQuery: "ALTER TABLE long_ord_pk2 MODIFY COLUMN y int AFTER u", ExpectedWriteResult: []sql.Row{ {sql.OkResult{RowsAffected: 0}}, }, SelectQuery: "SELECT column_name, ordinal_position FROM INFORMATION_SCHEMA.COLUMNS WHERE TABLE_SCHEMA = 'mydb' AND TABLE_NAME = 'long_ord_pk2' and column_key = 'PRI'", ExpectedSelect: []sql.Row{ {"u", uint(1)}, {"y", uint(2)}, {"v", uint(3)}, {"x", uint(5)}, {"z", uint(6)}, }, }, { WriteQuery: "ALTER TABLE long_ord_pk2 MODIFY COLUMN y int AFTER u", ExpectedWriteResult: []sql.Row{ {sql.OkResult{RowsAffected: 0}}, }, SelectQuery: "show keys from long_ord_pk2", ExpectedSelect: []sql.Row{ {"long_ord_pk2", 0, "PRIMARY", 1, "y", nil, 0, nil, nil, "", "BTREE", "", "", "YES", nil}, {"long_ord_pk2", 0, "PRIMARY", 2, "v", nil, 0, nil, nil, "", "BTREE", "", "", "YES", nil}, {"long_ord_pk2", 0, "PRIMARY", 3, "x", nil, 0, nil, nil, "", "BTREE", "", "", "YES", nil}, {"long_ord_pk2", 0, "PRIMARY", 4, "z", nil, 0, nil, nil, "", "BTREE", "", "", "YES", nil}, {"long_ord_pk2", 0, "PRIMARY", 5, "u", nil, 0, nil, nil, "", "BTREE", "", "", "YES", nil}, }, }, }
var PlanTests = []QueryPlanTest{}/* 221 elements not displayed */
PlanTests is a test of generating the right query plans for different queries in the presence of indexes and other features. These tests are fragile because they rely on string representations of query plans, but they're much easier to construct this way.
var ProcedureCallTests = []ScriptTest{ { Name: "OUT param with SET", SetUpScript: []string{ "SET @outparam = 5", "CREATE PROCEDURE testabc(OUT x BIGINT) SET x = 9", "CALL testabc(@outparam)", }, Assertions: []ScriptTestAssertion{ { Query: "SELECT @outparam", Expected: []sql.Row{ { int64(9), }, }, }, }, }, { Name: "OUT param without SET", SetUpScript: []string{ "SET @outparam = 5", "CREATE PROCEDURE testabc(OUT x BIGINT) SELECT x", "CALL testabc(@outparam)", }, Assertions: []ScriptTestAssertion{ { Query: "SELECT @outparam", Expected: []sql.Row{ { nil, }, }, }, }, }, { Name: "INOUT param with SET", SetUpScript: []string{ "SET @outparam = 5", "CREATE PROCEDURE testabc(INOUT x BIGINT) BEGIN SET x = x + 1; SET x = x + 3; END;", "CALL testabc(@outparam)", }, Assertions: []ScriptTestAssertion{ { Query: "SELECT @outparam", Expected: []sql.Row{ { int64(9), }, }, }, }, }, { Name: "INOUT param without SET", SetUpScript: []string{ "SET @outparam = 5", "CREATE PROCEDURE testabc(INOUT x BIGINT) SELECT x", "CALL testabc(@outparam)", }, Assertions: []ScriptTestAssertion{ { Query: "SELECT @outparam", Expected: []sql.Row{ { int64(5), }, }, }, }, }, { Name: "Nested CALL with INOUT param", SetUpScript: []string{ "SET @outparam = 5", "CREATE PROCEDURE p3(INOUT z INT) BEGIN SET z = z * 111; END;", "CREATE PROCEDURE p2(INOUT y DOUBLE) BEGIN SET y = y + 4; CALL p3(y); END;", "CREATE PROCEDURE p1(INOUT x BIGINT) BEGIN SET x = 3; CALL p2(x); END;", "CALL p1(@outparam)", }, Assertions: []ScriptTestAssertion{ { Query: "SELECT @outparam", Expected: []sql.Row{ { int64(777), }, }, }, }, }, { Name: "OUT param without SET", SetUpScript: []string{ "SET @outparam = 5", "CREATE PROCEDURE testabc(OUT x BIGINT) SELECT x", "CALL testabc(@outparam)", }, Assertions: []ScriptTestAssertion{ { Query: "SELECT @outparam", Expected: []sql.Row{ { nil, }, }, }, }, }, { Name: "Incompatible type for parameter", SetUpScript: []string{ "CREATE PROCEDURE p1(x DATETIME) SELECT x", }, Assertions: []ScriptTestAssertion{ { Query: "CALL p1('hi')", ExpectedErr: sql.ErrConvertingToTime, }, }, }, { Name: "Incorrect parameter count", SetUpScript: []string{ "CREATE PROCEDURE p1(x BIGINT, y BIGINT) SELECT x + y", }, Assertions: []ScriptTestAssertion{ { Query: "CALL p1(1)", ExpectedErr: sql.ErrCallIncorrectParameterCount, }, { Query: "CALL p1(1, 2, 3)", ExpectedErr: sql.ErrCallIncorrectParameterCount, }, }, }, { Name: "use procedure parameter in filter expressions and multiple statements", SetUpScript: []string{ "CREATE TABLE inventory (store_id int, product varchar(5))", "INSERT INTO inventory VALUES (1, 'a'), (1, 'b'), (1, 'c'), (1, 'd'), (2, 'e'), (2, 'f'), (1, 'g'), (1, 'h'), (3, 'i')", "CREATE PROCEDURE proc1 (IN p_store_id INT) SELECT COUNT(*) FROM inventory WHERE store_id = p_store_id;", "CREATE PROCEDURE proc2 (IN p_store_id INT, OUT p_film_count INT) READS SQL DATA BEGIN SELECT COUNT(*) as counted FROM inventory WHERE store_id = p_store_id; SET p_film_count = 44; END ;", }, Assertions: []ScriptTestAssertion{ { Query: "CALL proc1(1)", Expected: []sql.Row{ { int64(6), }, }, }, { Query: "CALL proc1(2)", Expected: []sql.Row{ { int64(2), }, }, }, { Query: "CALL proc1(4)", Expected: []sql.Row{ { int64(0), }, }, }, { Query: "CALL proc2(3, @foo)", Expected: []sql.Row{ { int64(1), }, }, }, { Query: "SELECT @foo", Expected: []sql.Row{ { int64(44), }, }, }, }, }, }
var ProcedureDropTests = []ScriptTest{ { Name: "DROP procedures", SetUpScript: []string{ "CREATE PROCEDURE p1() SELECT 5", "CREATE PROCEDURE p2() SELECT 6", }, Assertions: []ScriptTestAssertion{ { Query: "CALL p1", Expected: []sql.Row{ { int64(5), }, }, }, { Query: "CALL p2", Expected: []sql.Row{ { int64(6), }, }, }, { Query: "DROP PROCEDURE p1", Expected: []sql.Row{}, }, { Query: "CALL p1", ExpectedErr: sql.ErrStoredProcedureDoesNotExist, }, { Query: "DROP PROCEDURE IF EXISTS p2", Expected: []sql.Row{}, }, { Query: "CALL p2", ExpectedErr: sql.ErrStoredProcedureDoesNotExist, }, { Query: "DROP PROCEDURE p3", ExpectedErr: sql.ErrStoredProcedureDoesNotExist, }, { Query: "DROP PROCEDURE IF EXISTS p4", Expected: []sql.Row{}, }, }, }, }
var ProcedureLogicTests = []ScriptTest{ { Name: "Simple SELECT", SetUpScript: []string{ "CREATE PROCEDURE testabc(x DOUBLE, y DOUBLE) SELECT x*y", }, Assertions: []ScriptTestAssertion{ { Query: "CALL testabc(2, 3)", Expected: []sql.Row{ { float64(6), }, }, }, { Query: "CALL testabc(9, 9.5)", Expected: []sql.Row{ { float64(85.5), }, }, }, }, }, { Name: "Multiple SELECTs", SetUpScript: []string{ "CREATE TABLE t1(pk VARCHAR(20) PRIMARY KEY)", "INSERT INTO t1 VALUES (3), (4), (50)", `CREATE PROCEDURE p1() BEGIN SELECT * FROM t1; UPDATE t1 SET pk = CONCAT(pk, '0'); SELECT * FROM t1; INSERT INTO t1 VALUES (1), (2); SELECT * FROM t1; REPLACE INTO t1 VALUES (1), (30); DELETE FROM t1 WHERE pk LIKE '%00'; END;`, }, Assertions: []ScriptTestAssertion{ { Query: "CALL p1()", Expected: []sql.Row{ {"1"}, {"2"}, {"30"}, {"40"}, {"500"}, }, }, { Query: "SELECT * FROM t1 ORDER BY 1", Expected: []sql.Row{ {"1"}, {"2"}, {"30"}, {"40"}, }, }, }, }, { Name: "IF/ELSE with 1 SELECT at end", SetUpScript: []string{ "SET @outparam = ''", `CREATE PROCEDURE p1(OUT s VARCHAR(200), N DOUBLE, m DOUBLE) BEGIN SET s = ''; IF n = m THEN SET s = 'equals'; ELSE IF n > m THEN SET s = 'greater'; ELSE SET s = 'less'; END IF; SET s = CONCAT('is ', s, ' than'); END IF; SET s = CONCAT(n, ' ', s, ' ', m, '.'); SELECT s; END;`, `CREATE PROCEDURE p2(s VARCHAR(200), N DOUBLE, m DOUBLE) BEGIN SET s = ''; IF n = m THEN SET s = 'equals'; ELSE IF n > m THEN SET s = 'greater'; ELSE SET s = 'less'; END IF; SET s = CONCAT('is ', s, ' than'); END IF; SET s = CONCAT(n, ' ', s, ' ', m, '.'); SELECT s; END;`, }, Assertions: []ScriptTestAssertion{ { Query: "CALL p1(@outparam, 1, 2)", Expected: []sql.Row{ { "1 is less than 2.", }, }, }, { Query: "SELECT @outparam", Expected: []sql.Row{ { "1 is less than 2.", }, }, }, { Query: "CALL p1(@outparam, null, 2)", Expected: []sql.Row{ { nil, }, }, }, { Query: "CALL p1(@outparam, 7, 4)", Expected: []sql.Row{ { "7 is greater than 4.", }, }, }, { Query: "SELECT @outparam", Expected: []sql.Row{ { "7 is greater than 4.", }, }, }, { Query: "CALL p1(@outparam, 5, 5)", Expected: []sql.Row{ { "5 equals 5.", }, }, }, { Query: "SELECT @outparam", Expected: []sql.Row{ { "5 equals 5.", }, }, }, { Query: "CALL p2(@outparam, 9, 3)", Expected: []sql.Row{ { "9 is greater than 3.", }, }, }, { Query: "SELECT @outparam", Expected: []sql.Row{ { "5 equals 5.", }, }, }, }, }, { Name: "IF/ELSE with nested SELECT in branches", SetUpScript: []string{ "CREATE TABLE t1(pk BIGINT PRIMARY KEY)", `CREATE PROCEDURE p1(x BIGINT) BEGIN DELETE FROM t1; IF x < 10 THEN IF x = 0 THEN SELECT 1000; ELSEIF x = 1 THEN SELECT 1001; ELSE INSERT INTO t1 VALUES (3), (4), (5); END IF; ELSEIF x < 20 THEN IF x = 10 THEN INSERT INTO t1 VALUES (1), (2), (6), (7); ELSEIF x = 11 THEN INSERT INTO t1 VALUES (8), (9), (10), (11), (12); SELECT * FROM t1; ELSE SELECT 2002; SELECT 2003; END IF; END IF; INSERT INTO t1 VALUES (1), (2); END;`, }, Assertions: []ScriptTestAssertion{ { Query: "CALL p1(0)", Expected: []sql.Row{ { int64(1000), }, }, }, { Query: "CALL p1(1)", Expected: []sql.Row{ { int64(1001), }, }, }, { Query: "CALL p1(2)", Expected: []sql.Row{ { sql.NewOkResult(2), }, }, }, { Query: "CALL p1(10)", ExpectedErr: sql.ErrPrimaryKeyViolation, }, { Query: "CALL p1(11)", Expected: []sql.Row{ {int64(8)}, {int64(9)}, {int64(10)}, {int64(11)}, {int64(12)}, }, }, { Query: "CALL p1(12)", Expected: []sql.Row{ { int64(2003), }, }, }, }, }, { Name: "SELECT with JOIN and table aliases", SetUpScript: []string{ "CREATE TABLE foo(a BIGINT PRIMARY KEY, b VARCHAR(20))", "INSERT INTO foo VALUES (1, 'd'), (2, 'e'), (3, 'f')", "CREATE TABLE bar(b VARCHAR(30) PRIMARY KEY, c BIGINT)", "INSERT INTO bar VALUES ('x', 3), ('y', 2), ('z', 1)", "CREATE PROCEDURE p1() SELECT f.a, bar.b, f.b FROM foo f INNER JOIN bar ON f.a = bar.c ORDER BY 1", "CREATE PROCEDURE p2() BEGIN SELECT f.a, bar.b, f.b FROM foo f INNER JOIN bar ON f.a = bar.c ORDER BY 1; END;", "CREATE PROCEDURE p3() IF 0 = 0 THEN SELECT f.a, bar.b, f.b FROM foo f INNER JOIN bar ON f.a = bar.c ORDER BY 1; END IF;", "CREATE PROCEDURE p4() BEGIN SELECT 7; SELECT f.a, bar.b, f.b FROM foo f INNER JOIN bar ON f.a = bar.c ORDER BY 1; END;", "CREATE PROCEDURE p5() IF 0 = 0 THEN SELECT 7; SELECT f.a, bar.b, f.b FROM foo f INNER JOIN bar ON f.a = bar.c ORDER BY 1; END IF;", }, Assertions: []ScriptTestAssertion{ { Query: "SELECT f.a, bar.b, f.b FROM foo f INNER JOIN bar ON f.a = bar.c ORDER BY 1", Expected: []sql.Row{ {int64(1), "z", "d"}, {int64(2), "y", "e"}, {int64(3), "x", "f"}, }, }, { Query: "CALL p1()", Expected: []sql.Row{ {int64(1), "z", "d"}, {int64(2), "y", "e"}, {int64(3), "x", "f"}, }, }, { Query: "CALL p2()", Expected: []sql.Row{ {int64(1), "z", "d"}, {int64(2), "y", "e"}, {int64(3), "x", "f"}, }, }, { Query: "CALL p3()", Expected: []sql.Row{ {int64(1), "z", "d"}, {int64(2), "y", "e"}, {int64(3), "x", "f"}, }, }, { Query: "CALL p4()", Expected: []sql.Row{ {int64(1), "z", "d"}, {int64(2), "y", "e"}, {int64(3), "x", "f"}, }, }, { Query: "CALL p5()", Expected: []sql.Row{ {int64(1), "z", "d"}, {int64(2), "y", "e"}, {int64(3), "x", "f"}, }, }, }, }, { Name: "Nested CALL in IF/ELSE branch", SetUpScript: []string{ "CREATE TABLE t1(pk BIGINT PRIMARY KEY)", "INSERT INTO t1 VALUES (2), (3)", "CREATE PROCEDURE p1(INOUT x BIGINT) BEGIN IF X = 1 THEN CALL p2(10); ELSEIF x = 2 THEN CALL p2(100); ELSE CALL p2(X); END IF; END;", "CREATE PROCEDURE p2(INOUT y BIGINT) BEGIN SELECT pk * y FROM t1 ORDER BY 1; END;", }, Assertions: []ScriptTestAssertion{ { Query: "CALL p1(1)", Expected: []sql.Row{ {int64(20)}, {int64(30)}, }, }, { Query: "CALL p1(2)", Expected: []sql.Row{ {int64(200)}, {int64(300)}, }, }, { Query: "CALL p1(5)", Expected: []sql.Row{ {int64(10)}, {int64(15)}, }, }, }, }, { Name: "INSERT INTO SELECT doesn't override SELECT", SetUpScript: []string{ "CREATE TABLE t1(pk BIGINT PRIMARY KEY)", "CREATE TABLE t2(pk BIGINT PRIMARY KEY)", "INSERT INTO t1 VALUES (2), (3)", "INSERT INTO t2 VALUES (1)", `CREATE PROCEDURE p1(x BIGINT) BEGIN DELETE FROM t2 WHERE pk > 1; INSERT INTO t2 SELECT pk FROM t1; SELECT * FROM t2; INSERT INTO t2 SELECT pk + 10 FROM t1; IF x = 1 THEN SELECT * FROM t2; END IF; END;`, }, Assertions: []ScriptTestAssertion{ { Query: "CALL p1(0)", Expected: []sql.Row{ {int64(1)}, {int64(2)}, {int64(3)}, }, }, { Query: "CALL p1(1)", Expected: []sql.Row{ {int64(1)}, {int64(2)}, {int64(3)}, {int64(12)}, {int64(13)}, }, }, }, }, { Name: "Parameters resolve inside of INSERT", SetUpScript: []string{ `CREATE TABLE items ( id INT PRIMARY KEY AUTO_INCREMENT, item TEXT NOT NULL );`, `CREATE PROCEDURE add_item (IN txt TEXT) MODIFIES SQL DATA INSERT INTO items (item) VALUES (txt)`, }, Assertions: []ScriptTestAssertion{ { Query: "CALL add_item('A test item');", Expected: []sql.Row{ {sql.OkResult{RowsAffected: 1, InsertID: 1}}, }, }, { Query: "SELECT * FROM items;", Expected: []sql.Row{ {1, "A test item"}, }, }, }, }, { Name: "Parameters resolve inside of SELECT UNION", SetUpScript: []string{ "CREATE TABLE t1(pk BIGINT PRIMARY KEY, v1 BIGINT)", "INSERT INTO t1 VALUES (1, 2)", "SELECT pk, v1 FROM t1 UNION SELECT 1, 2;", `CREATE PROCEDURE p1(x BIGINT, y BIGINT) BEGIN SELECT pk+x, v1+y FROM t1 UNION SELECT x, y; END;`, `CREATE PROCEDURE p2(u BIGINT, v BIGINT) SELECT pk+u, v1+v FROM t1 UNION SELECT u, v;`, }, Assertions: []ScriptTestAssertion{ { Query: "CALL p1(3, 4)", Expected: []sql.Row{ {"4", "6"}, {"3", "4"}, }, }, { Query: "CALL p2(5, 6)", Expected: []sql.Row{ {"6", "8"}, {"5", "6"}, }, }, }, }, { Name: "Parameters resolve inside of REPLACE", SetUpScript: []string{ `CREATE TABLE items ( id INT PRIMARY KEY AUTO_INCREMENT, item INT NOT NULL );`, `CREATE PROCEDURE add_item (IN num INT) MODIFIES SQL DATA BEGIN REPLACE INTO items (item) VALUES (5), (num), (num+1); END`, }, Assertions: []ScriptTestAssertion{ { Query: "CALL add_item(6);", Expected: []sql.Row{ {sql.NewOkResult(3)}, }, }, { Query: "SELECT * FROM items ORDER BY 1;", Expected: []sql.Row{ {1, 5}, {2, 6}, {3, 7}, }, }, }, }, { Name: "Parameters resolve inside of INSERT INTO SELECT", SetUpScript: []string{ "CREATE TABLE t1(pk BIGINT PRIMARY KEY)", "CREATE TABLE t2(pk BIGINT PRIMARY KEY)", "INSERT INTO t1 VALUES (1), (2)", `CREATE PROCEDURE p1(x BIGINT) BEGIN TRUNCATE t2; INSERT INTO t2 SELECT pk+x FROM t1; SELECT * FROM t2; END;`, }, Assertions: []ScriptTestAssertion{ { Query: "CALL p1(0)", Expected: []sql.Row{ {int64(1)}, {int64(2)}, }, }, { Query: "CALL p1(5)", Expected: []sql.Row{ {int64(6)}, {int64(7)}, }, }, }, }, { Name: "Subquery on SET user variable captures parameter", SetUpScript: []string{ `CREATE PROCEDURE p1(x VARCHAR(20)) BEGIN SET @randomvar = (SELECT LENGTH(x)); SELECT @randomvar; END;`, }, Assertions: []ScriptTestAssertion{ { Query: "CALL p1('hi')", Expected: []sql.Row{ {int64(2)}, }, }, { Query: "CALL p1('hello')", Expected: []sql.Row{ {int64(5)}, }, }, }, }, { Name: "Simple SELECT INTO", SetUpScript: []string{ "CREATE PROCEDURE testabc(IN x DOUBLE, IN y DOUBLE, OUT abc DOUBLE) SELECT x*y INTO abc", "CALL testabc(2, 3, @res1)", "CALL testabc(9, 9.5, @res2)", }, Assertions: []ScriptTestAssertion{ { Query: "SELECT @res1", Expected: []sql.Row{{float64(6)}}, }, { Query: "SELECT @res2", Expected: []sql.Row{{float64(85.5)}}, }, }, }, { Name: "Multiple variables in SELECT INTO", SetUpScript: []string{ "CREATE PROCEDURE new_proc(IN x DOUBLE, IN y DOUBLE, OUT abc DOUBLE, OUT def DOUBLE) SELECT x*y, x+y INTO abc, def", "CALL new_proc(2, 3, @res1, @res2)", "CALL new_proc(9, 9.5, @res3, @res4)", }, Assertions: []ScriptTestAssertion{ { Query: "SELECT @res1, @res2", Expected: []sql.Row{{float64(6), float64(5)}}, }, { Query: "SELECT @res3, @res4", Expected: []sql.Row{{float64(85.5), float64(18.5)}}, }, }, }, { Name: "SELECT INTO with condition", SetUpScript: []string{ "CREATE TABLE inventory (item_id int primary key, shelf_id int, items varchar(100))", "INSERT INTO inventory VALUES (1, 1, 'a'), (2, 1, 'b'), (3, 2, 'c'), (4, 1, 'd'), (5, 4, 'e')", "CREATE PROCEDURE in_stock (IN p_id INT, OUT p_count INT) SELECT COUNT(*) FROM inventory WHERE shelf_id = p_id INTO p_count", "CALL in_stock(1, @shelf1)", "CALL in_stock(2, @shelf2)", "CALL in_stock(3, @shelf3)", }, Assertions: []ScriptTestAssertion{ { Query: "SELECT @shelf1, @shelf2, @shelf3", Expected: []sql.Row{{3, 1, 0}}, }, }, }, { Name: "SELECT INTO with group by, order by and limit", SetUpScript: []string{ "CREATE TABLE inventory (item_id int primary key, shelf_id int, item varchar(10))", "INSERT INTO inventory VALUES (1, 1, 'a'), (2, 1, 'b'), (3, 2, 'c'), (4, 1, 'd'), (5, 4, 'e')", "CREATE PROCEDURE first_shelf (OUT p_count INT) SELECT COUNT(*) FROM inventory GROUP BY shelf_id ORDER BY shelf_id ASC LIMIT 1 INTO p_count", "CREATE PROCEDURE last_shelf (OUT p_count INT) SELECT COUNT(*) FROM inventory GROUP BY shelf_id ORDER BY shelf_id DESC LIMIT 1 INTO p_count", "CALL first_shelf(@result1)", "CALL last_shelf(@result2)", }, Assertions: []ScriptTestAssertion{ { Query: "SELECT @result1", Expected: []sql.Row{{3}}, }, { Query: "SELECT @result2", Expected: []sql.Row{{1}}, }, }, }, { Name: "multiple SELECT INTO in begin end block", SetUpScript: []string{ "CREATE TABLE inventory (item_id int primary key, shelf_id int, item varchar(10))", "INSERT INTO inventory VALUES (1, 1, 'a'), (2, 1, 'b'), (3, 2, 'c'), (4, 1, 'd'), (5, 4, 'e')", "CREATE PROCEDURE random_info(OUT p_count1 INT, OUT p_count2 VARCHAR(10)) BEGIN " + "SELECT COUNT(*) FROM inventory GROUP BY shelf_id ORDER BY shelf_id ASC LIMIT 1 INTO p_count1;" + "SELECT item INTO p_count2 FROM inventory WHERE shelf_id = 1 ORDER BY item DESC LIMIT 1; " + "END", "CALL random_info(@s1, @s2)", }, Assertions: []ScriptTestAssertion{ { Query: "SELECT @s1, @s2", Expected: []sql.Row{{3, "d"}}, }, }, }, { Name: "multiple statement with single SELECT INTO in begin end block", SetUpScript: []string{ "CREATE TABLE inventory (item_id int primary key, shelf_id int, item varchar(10))", "INSERT INTO inventory VALUES (1, 1, 'a'), (2, 1, 'b'), (3, 2, 'c'), (4, 1, 'd'), (5, 4, 'e')", `CREATE PROCEDURE count_and_print(IN p_shelf_id INT, OUT p_count INT) BEGIN SELECT item FROM inventory WHERE shelf_id = p_shelf_id ORDER BY item ASC; SELECT COUNT(*) INTO p_count FROM inventory WHERE shelf_id = p_shelf_id; END`, }, Assertions: []ScriptTestAssertion{ { Query: "CALL count_and_print(1, @total)", Expected: []sql.Row{{"a"}, {"b"}, {"d"}}, }, { Query: "SELECT @total", Expected: []sql.Row{{3}}, }, }, }, { Name: "DECLARE CONDITION", SetUpScript: []string{ `CREATE PROCEDURE p1(x INT) BEGIN DECLARE specialty CONDITION FOR SQLSTATE '45000'; DECLARE specialty2 CONDITION FOR SQLSTATE '02000'; IF x = 0 THEN SIGNAL SQLSTATE '01000'; ELSEIF x = 1 THEN SIGNAL SQLSTATE '45000' SET MESSAGE_TEXT = 'A custom error occurred 1'; ELSEIF x = 2 THEN SIGNAL specialty SET MESSAGE_TEXT = 'A custom error occurred 2', MYSQL_ERRNO = 1002; ELSEIF x = 3 THEN SIGNAL specialty; ELSEIF x = 4 THEN SIGNAL specialty2; ELSE SIGNAL SQLSTATE '01000' SET MESSAGE_TEXT = 'A warning occurred', MYSQL_ERRNO = 1000; SIGNAL SQLSTATE '45000' SET MESSAGE_TEXT = 'An error occurred', MYSQL_ERRNO = 1001; END IF; BEGIN DECLARE specialty3 CONDITION FOR SQLSTATE '45000'; END; END;`, }, Assertions: []ScriptTestAssertion{ { Query: "CALL p1(0)", ExpectedErrStr: "warnings not yet implemented", }, { Query: "CALL p1(1)", ExpectedErrStr: "A custom error occurred 1 (errno 1644) (sqlstate 45000)", }, { Query: "CALL p1(2)", ExpectedErrStr: "A custom error occurred 2 (errno 1002) (sqlstate 45000)", }, { Query: "CALL p1(3)", ExpectedErrStr: "Unhandled user-defined exception condition (errno 1644) (sqlstate 45000)", }, { Query: "CALL p1(4)", ExpectedErrStr: "Unhandled user-defined not found condition (errno 1643) (sqlstate 02000)", }, }, }, { Name: "DECLARE CONDITION nesting priority", SetUpScript: []string{ `CREATE PROCEDURE p1(x INT) BEGIN DECLARE cond_name CONDITION FOR SQLSTATE '02000'; BEGIN DECLARE cond_name CONDITION FOR SQLSTATE '45000'; IF x = 0 THEN SIGNAL cond_name; END IF; END; SIGNAL cond_name; END;`, `CREATE PROCEDURE p2(x INT) BEGIN DECLARE cond_name CONDITION FOR SQLSTATE '45000'; BEGIN DECLARE cond_name CONDITION FOR SQLSTATE '02000'; IF x = 0 THEN SIGNAL cond_name; END IF; END; SIGNAL cond_name; END;`, }, Assertions: []ScriptTestAssertion{ { Query: "CALL p1(0)", ExpectedErrStr: "Unhandled user-defined exception condition (errno 1644) (sqlstate 45000)", }, { Query: "CALL p1(1)", ExpectedErrStr: "Unhandled user-defined not found condition (errno 1643) (sqlstate 02000)", }, { Query: "CALL p2(0)", ExpectedErrStr: "Unhandled user-defined not found condition (errno 1643) (sqlstate 02000)", }, { Query: "CALL p2(1)", ExpectedErrStr: "Unhandled user-defined exception condition (errno 1644) (sqlstate 45000)", }, }, }, { Name: "Duplicate parameter names", Query: "CREATE PROCEDURE p1(abc DATETIME, abc DOUBLE) SELECT abc", ExpectedErr: sql.ErrProcedureDuplicateParameterName, }, { Name: "Duplicate parameter names mixed casing", Query: "CREATE PROCEDURE p1(abc DATETIME, ABC DOUBLE) SELECT abc", ExpectedErr: sql.ErrProcedureDuplicateParameterName, }, { Name: "Invalid parameter type", Query: "CREATE PROCEDURE p1(x FAKETYPE) SELECT x", ExpectedErr: sql.ErrSyntaxError, }, { Name: "Invalid USE statement", Query: `CREATE PROCEDURE p1() USE mydb`, ExpectedErr: sql.ErrSyntaxError, }, { Name: "Invalid LOCK/UNLOCK statements", SetUpScript: []string{ "CREATE TABLE t1(pk BIGINT PRIMARY KEY)", }, Assertions: []ScriptTestAssertion{ { Query: "CREATE PROCEDURE p1(x BIGINT) LOCK TABLES t1 READ", ExpectedErr: sql.ErrSyntaxError, }, { Query: "CREATE PROCEDURE p1(x BIGINT) UNLOCK TABLES", ExpectedErr: sql.ErrSyntaxError, }, }, }, { Name: "DECLARE CONDITION wrong positions", Assertions: []ScriptTestAssertion{ { Query: `CREATE PROCEDURE p1(x INT) BEGIN SELECT x; DECLARE cond_name CONDITION FOR SQLSTATE '45000'; END;`, ExpectedErr: sql.ErrDeclareOrderInvalid, }, { Query: `CREATE PROCEDURE p1(x INT) BEGIN BEGIN SELECT x; DECLARE cond_name CONDITION FOR SQLSTATE '45000'; END; END;`, ExpectedErr: sql.ErrDeclareOrderInvalid, }, { Query: `CREATE PROCEDURE p1(x INT) BEGIN IF x = 0 THEN DECLARE cond_name CONDITION FOR SQLSTATE '45000'; END IF; END;`, ExpectedErr: sql.ErrDeclareOrderInvalid, }, { Query: `CREATE PROCEDURE p1(x INT) BEGIN IF x = 0 THEN SELECT x; ELSE DECLARE cond_name CONDITION FOR SQLSTATE '45000'; END IF; END;`, ExpectedErr: sql.ErrDeclareOrderInvalid, }, }, }, { Name: "DECLARE CONDITION duplicate name", Query: `CREATE PROCEDURE p1() BEGIN DECLARE cond_name CONDITION FOR SQLSTATE '45000'; DECLARE cond_name CONDITION FOR SQLSTATE '45000'; END;`, ExpectedErr: sql.ErrDeclareConditionDuplicate, }, { Name: "SIGNAL references condition name for MySQL error code", Query: `CREATE PROCEDURE p1(x INT) BEGIN DECLARE mysql_err_code CONDITION FOR 1000; SIGNAL mysql_err_code; END;`, ExpectedErr: sql.ErrUnsupportedSyntax, }, { Name: "SIGNAL non-existent condition name", Query: `CREATE PROCEDURE p1(x INT) BEGIN DECLARE abcdefg CONDITION FOR SQLSTATE '45000'; SIGNAL abcdef; END;`, ExpectedErr: sql.ErrDeclareConditionNotFound, }, { Name: "Duplicate procedure name", SetUpScript: []string{ "CREATE PROCEDURE test_proc(x DOUBLE, y DOUBLE) SELECT x*y", }, Query: "CREATE PROCEDURE test_proc(z VARCHAR(20)) SELECT z", ExpectedErr: sql.ErrStoredProcedureAlreadyExists, }, }
var ProcedureShowCreate = []ScriptTest{ { Name: "SHOW procedures", SetUpScript: []string{ "CREATE PROCEDURE p1() COMMENT 'hi' DETERMINISTIC SELECT 6", "CREATE definer=`user` PROCEDURE p2() SQL SECURITY INVOKER SELECT 7", "CREATE PROCEDURE p21() SQL SECURITY DEFINER SELECT 8", }, Assertions: []ScriptTestAssertion{ { Query: "SHOW CREATE PROCEDURE p1", Expected: []sql.Row{ { "p1", "", "CREATE PROCEDURE p1() COMMENT 'hi' DETERMINISTIC SELECT 6", "utf8mb4", "utf8mb4_0900_bin", "utf8mb4_0900_bin", }, }, }, { Query: "SHOW CREATE PROCEDURE p2", Expected: []sql.Row{ { "p2", "", "CREATE definer=`user` PROCEDURE p2() SQL SECURITY INVOKER SELECT 7", "utf8mb4", "utf8mb4_0900_bin", "utf8mb4_0900_bin", }, }, }, { Query: "SHOW CREATE PROCEDURE p21", Expected: []sql.Row{ { "p21", "", "CREATE PROCEDURE p21() SQL SECURITY DEFINER SELECT 8", "utf8mb4", "utf8mb4_0900_bin", "utf8mb4_0900_bin", }, }, }, }, }, { Name: "SHOW non-existent procedures", SetUpScript: []string{}, Assertions: []ScriptTestAssertion{ { Query: "SHOW CREATE PROCEDURE p1", ExpectedErr: sql.ErrStoredProcedureDoesNotExist, }, }, }, }
var ProcedureShowStatus = []ScriptTest{ { Name: "SHOW procedures", SetUpScript: []string{ "CREATE PROCEDURE p1() COMMENT 'hi' DETERMINISTIC SELECT 6", "CREATE definer=`user` PROCEDURE p2() SQL SECURITY INVOKER SELECT 7", "CREATE PROCEDURE p21() SQL SECURITY DEFINER SELECT 8", }, Assertions: []ScriptTestAssertion{ { Query: "SHOW PROCEDURE STATUS", Expected: []sql.Row{ { "mydb", "p1", "PROCEDURE", "", time.Unix(0, 0).UTC(), time.Unix(0, 0).UTC(), "DEFINER", "hi", "utf8mb4", "utf8mb4_0900_bin", "utf8mb4_0900_bin", }, { "mydb", "p2", "PROCEDURE", "`user`@`%`", time.Unix(0, 0).UTC(), time.Unix(0, 0).UTC(), "INVOKER", "", "utf8mb4", "utf8mb4_0900_bin", "utf8mb4_0900_bin", }, { "mydb", "p21", "PROCEDURE", "", time.Unix(0, 0).UTC(), time.Unix(0, 0).UTC(), "DEFINER", "", "utf8mb4", "utf8mb4_0900_bin", "utf8mb4_0900_bin", }, }, }, { Query: "SHOW PROCEDURE STATUS LIKE 'p2%'", Expected: []sql.Row{ { "mydb", "p2", "PROCEDURE", "`user`@`%`", time.Unix(0, 0).UTC(), time.Unix(0, 0).UTC(), "INVOKER", "", "utf8mb4", "utf8mb4_0900_bin", "utf8mb4_0900_bin", }, { "mydb", "p21", "PROCEDURE", "", time.Unix(0, 0).UTC(), time.Unix(0, 0).UTC(), "DEFINER", "", "utf8mb4", "utf8mb4_0900_bin", "utf8mb4_0900_bin", }, }, }, { Query: "SHOW PROCEDURE STATUS LIKE 'p4'", Expected: []sql.Row{}, }, { Query: "SHOW PROCEDURE STATUS WHERE Db = 'mydb'", Expected: []sql.Row{ { "mydb", "p1", "PROCEDURE", "", time.Unix(0, 0).UTC(), time.Unix(0, 0).UTC(), "DEFINER", "hi", "utf8mb4", "utf8mb4_0900_bin", "utf8mb4_0900_bin", }, { "mydb", "p2", "PROCEDURE", "`user`@`%`", time.Unix(0, 0).UTC(), time.Unix(0, 0).UTC(), "INVOKER", "", "utf8mb4", "utf8mb4_0900_bin", "utf8mb4_0900_bin", }, { "mydb", "p21", "PROCEDURE", "", time.Unix(0, 0).UTC(), time.Unix(0, 0).UTC(), "DEFINER", "", "utf8mb4", "utf8mb4_0900_bin", "utf8mb4_0900_bin", }, }, }, { Query: "SHOW PROCEDURE STATUS WHERE Name LIKE '%1'", Expected: []sql.Row{ { "mydb", "p1", "PROCEDURE", "", time.Unix(0, 0).UTC(), time.Unix(0, 0).UTC(), "DEFINER", "hi", "utf8mb4", "utf8mb4_0900_bin", "utf8mb4_0900_bin", }, { "mydb", "p21", "PROCEDURE", "", time.Unix(0, 0).UTC(), time.Unix(0, 0).UTC(), "DEFINER", "", "utf8mb4", "utf8mb4_0900_bin", "utf8mb4_0900_bin", }, }, }, { Query: "SHOW PROCEDURE STATUS WHERE Security_type = 'INVOKER'", Expected: []sql.Row{ { "mydb", "p2", "PROCEDURE", "`user`@`%`", time.Unix(0, 0).UTC(), time.Unix(0, 0).UTC(), "INVOKER", "", "utf8mb4", "utf8mb4_0900_bin", "utf8mb4_0900_bin", }, }, }, }, }, }
var QueryPlanTODOs = []QueryPlanTest{
{
Query: `SELECT pk,i,f FROM one_pk RIGHT JOIN niltable ON pk=i and pk > 0 ORDER BY 2,3`,
ExpectedPlan: "Sort(niltable.i ASC, niltable.f ASC)\n" +
" └─ Project(one_pk.pk, niltable.i, niltable.f)\n" +
" └─ RightJoin((one_pk.pk = niltable.i) AND (one_pk.pk > 0))\n" +
" ├─ Projected table access on [pk]\n" +
" │ └─ Table(one_pk)\n" +
" └─ Projected table access on [i f]\n" +
" └─ Table(niltable)\n" +
"",
},
{
Query: `SELECT pk,pk1,pk2 FROM one_pk t1, two_pk t2 WHERE pk=1 AND pk2=1 AND pk1=1 ORDER BY 1,2`,
ExpectedPlan: "Sort(t1.pk ASC, t2.pk1 ASC)\n" +
" └─ Project(t1.pk, t2.pk1, t2.pk2)\n" +
" └─ CrossJoin\n" +
" ├─ Filter(t1.pk = 1)\n" +
" │ └─ TableAlias(t1)\n" +
" │ └─ Projected table access on [pk]\n" +
" │ └─ IndexedTableAccess(one_pk on [one_pk.pk])\n" +
" └─ Filter((t2.pk2 = 1) AND (t2.pk1 = 1))\n" +
" └─ TableAlias(t2)\n" +
" └─ Table(two_pk)\n" +
"",
},
}
Queries where the query planner produces a correct (results) but suboptimal plan.
var QueryTests = []QueryTest{}/* 992 elements not displayed */
var QuickPrivTests = []QuickPrivilegeTest{ { Queries: []string{ "GRANT SELECT ON *.* TO tester@localhost", "SELECT * FROM mydb.test", }, Expected: []sql.Row{{0, 0}, {1, 1}}, }, { Queries: []string{ "GRANT SELECT ON mydb.* TO tester@localhost", "SELECT * FROM mydb.test", }, Expected: []sql.Row{{0, 0}, {1, 1}}, }, { Queries: []string{ "GRANT SELECT ON mydb.* TO tester@localhost", "SELECT * FROM mydb.test2", }, Expected: []sql.Row{{0, 1}, {1, 2}}, }, { Queries: []string{ "GRANT SELECT ON mydb.test TO tester@localhost", "SELECT * FROM mydb.test", }, Expected: []sql.Row{{0, 0}, {1, 1}}, }, { Queries: []string{ "GRANT SELECT ON mydb.test TO tester@localhost", "SELECT * FROM mydb.test2", }, ExpectingErr: true, }, { Queries: []string{ "GRANT SELECT ON otherdb.* TO tester@localhost", "SELECT * FROM mydb.test", }, ExpectingErr: true, }, { Queries: []string{ "GRANT SELECT ON otherdb.test TO tester@localhost", "SELECT * FROM mydb.test", }, ExpectingErr: true, }, { Queries: []string{ "GRANT SELECT ON otherdb.test TO tester@localhost", "SELECT * FROM mydb.test", }, ExpectingErr: true, }, { Queries: []string{ "GRANT SELECT ON *.* TO tester@localhost", "USE mydb;", "SHOW TABLES;", }, Expected: []sql.Row{{"test"}, {"test2"}}, }, { Queries: []string{ "GRANT SELECT ON mydb.* TO tester@localhost", "USE mydb;", "SHOW TABLES;", }, Expected: []sql.Row{{"test"}, {"test2"}}, }, { Queries: []string{ "GRANT SELECT ON mydb.test TO tester@localhost", "USE mydb;", "SHOW TABLES;", }, Expected: []sql.Row{{"test"}}, }, { Queries: []string{ "GRANT SELECT ON mydb.non_exist TO tester@localhost", "USE mydb;", "SHOW TABLES;", }, Expected: []sql.Row{}, }, { Queries: []string{ "ALTER TABLE mydb.test ADD COLUMN new_column BIGINT;", }, ExpectingErr: true, }, { Queries: []string{ "GRANT ALTER ON *.* TO tester@localhost", "ALTER TABLE mydb.test ADD COLUMN new_column BIGINT", }, }, { Queries: []string{ "GRANT ALTER ON mydb.* TO tester@localhost", "ALTER TABLE mydb.test ADD COLUMN new_column BIGINT;", }, }, { Queries: []string{ "GRANT ALTER ON mydb.test TO tester@localhost", "ALTER TABLE mydb.test ADD COLUMN new_column BIGINT;", }, }, { Queries: []string{ "ALTER TABLE mydb.test RENAME TO mydb.new_test;", }, ExpectingErr: true, }, { Queries: []string{ "GRANT ALTER ON *.* TO tester@localhost", "ALTER TABLE mydb.test RENAME TO mydb.new_test;", }, ExpectingErr: true, }, { Queries: []string{ "GRANT ALTER, CREATE, DROP, INSERT ON *.* TO tester@localhost", "ALTER TABLE mydb.test RENAME TO mydb.new_test;", }, }, { Queries: []string{ "GRANT ALTER, CREATE, DROP, INSERT ON mydb.* TO tester@localhost", "ALTER TABLE mydb.test RENAME TO mydb.new_test;", }, }, { Queries: []string{ "GRANT ALTER, CREATE, DROP, INSERT ON mydb.test TO tester@localhost", "ALTER TABLE mydb.test RENAME TO mydb.new_test;", }, ExpectingErr: true, }, { Queries: []string{ "GRANT ALTER, DROP ON mydb.test TO tester@localhost", "GRANT CREATE, INSERT ON mydb.new_test TO tester@localhost", "ALTER TABLE mydb.test RENAME TO mydb.new_test;", }, }, { Queries: []string{ "GRANT ALTER ON mydb.test TO tester@localhost", "GRANT CREATE, INSERT ON mydb.new_test TO tester@localhost", "ALTER TABLE mydb.test RENAME TO mydb.new_test;", }, ExpectingErr: true, }, { Queries: []string{ "GRANT DROP ON mydb.test TO tester@localhost", "GRANT CREATE, INSERT ON mydb.new_test TO tester@localhost", "ALTER TABLE mydb.test RENAME TO mydb.new_test;", }, ExpectingErr: true, }, { Queries: []string{ "GRANT ALTER, DROP ON mydb.test TO tester@localhost", "GRANT CREATE ON mydb.new_test TO tester@localhost", "ALTER TABLE mydb.test RENAME TO mydb.new_test;", }, ExpectingErr: true, }, { Queries: []string{ "GRANT ALTER, DROP ON mydb.test TO tester@localhost", "GRANT INSERT ON mydb.new_test TO tester@localhost", "ALTER TABLE mydb.test RENAME TO mydb.new_test;", }, ExpectingErr: true, }, { Queries: []string{ "USE mydb;", "CREATE PROCEDURE new_proc (x DOUBLE, y DOUBLE) SELECT x*y;", "DROP PROCEDURE new_proc;", }, ExpectingErr: true, }, { Queries: []string{ "GRANT ALTER ROUTINE ON *.* TO tester@localhost", "USE mydb;", "CREATE PROCEDURE new_proc (x DOUBLE, y DOUBLE) SELECT x*y;", "DROP PROCEDURE new_proc;", }, }, { Queries: []string{ "GRANT ALTER ROUTINE ON mydb.* TO tester@localhost", "USE mydb;", "CREATE PROCEDURE new_proc (x DOUBLE, y DOUBLE) SELECT x*y;", "DROP PROCEDURE new_proc;", }, }, { Queries: []string{ "CREATE DATABASE new_db;", }, ExpectingErr: true, }, { Queries: []string{ "CREATE TABLE mydb.new_table (pk BIGINT PRIMARY KEY);", }, ExpectingErr: true, }, { Queries: []string{ "GRANT CREATE ON *.* TO tester@localhost", "CREATE DATABASE new_db2;", "GRANT DROP ON *.* TO tester@localhost", "drop database new_db2", }, }, { Queries: []string{ "GRANT CREATE ON *.* TO tester@localhost", "CREATE TABLE mydb.new_table (pk BIGINT PRIMARY KEY);", }, }, { Queries: []string{ "GRANT CREATE ON mydb.* TO tester@localhost", "CREATE DATABASE new_db3;", "GRANT DROP ON *.* TO tester@localhost", "drop database new_db3", }, }, { Queries: []string{ "GRANT CREATE ON mydb.* TO tester@localhost", "CREATE TABLE mydb.new_table (pk BIGINT PRIMARY KEY);", }, }, { Queries: []string{ "CREATE ROLE new_role;", }, ExpectingErr: true, }, { Queries: []string{ "GRANT CREATE ROLE ON *.* TO tester@localhost", "CREATE ROLE new_role;", }, }, { Queries: []string{ "USE mydb;", "CREATE PROCEDURE new_proc (x DOUBLE, y DOUBLE) SELECT x*y;", }, ExpectingErr: true, }, { Queries: []string{ "GRANT CREATE ROUTINE ON *.* TO tester@localhost", "USE mydb;", "CREATE PROCEDURE new_proc (x DOUBLE, y DOUBLE) SELECT x*y;", }, }, { Queries: []string{ "GRANT CREATE ROUTINE ON mydb.* TO tester@localhost", "USE mydb;", "CREATE PROCEDURE new_proc (x DOUBLE, y DOUBLE) SELECT x*y;", }, }, { Queries: []string{ "CREATE USER new_user;", }, ExpectingErr: true, }, { Queries: []string{ "CREATE USER new_user;", "DROP USER new_user;", }, ExpectingErr: true, }, { Queries: []string{ "GRANT CREATE USER ON *.* TO tester@localhost", "CREATE USER new_user;", }, }, { Queries: []string{ "GRANT CREATE USER ON *.* TO tester@localhost", "CREATE USER new_user;", "DROP USER new_user;", }, }, { Queries: []string{ "GRANT CREATE USER ON *.* TO tester@localhost", "CREATE ROLE new_role;", }, }, { Queries: []string{ "GRANT CREATE USER ON *.* TO tester@localhost", "CREATE ROLE new_role;", "DROP ROLE new_role;", }, }, { Queries: []string{ "CREATE VIEW new_view AS SELECT 1;", }, ExpectingErr: true, }, { Queries: []string{ "GRANT CREATE VIEW ON *.* TO tester@localhost", "CREATE VIEW new_view AS SELECT 1;", }, }, { Queries: []string{ "DELETE FROM mydb.test WHERE pk >= 0;", }, ExpectingErr: true, }, { Queries: []string{ "GRANT DELETE ON *.* TO tester@localhost", "DELETE FROM mydb.test WHERE pk >= 0;", }, }, { Queries: []string{ "GRANT DELETE ON mydb.* TO tester@localhost", "DELETE FROM mydb.test WHERE pk >= 0;", }, }, { Queries: []string{ "GRANT DELETE ON mydb.test TO tester@localhost", "DELETE FROM mydb.test WHERE pk >= 0;", }, }, { Queries: []string{ "CREATE DATABASE new_db4;", }, ExpectingErr: true, }, { Queries: []string{ "CREATE TABLE mydb.new_table (pk BIGINT PRIMARY KEY);", "DROP TABLE mydb.new_table;", }, ExpectingErr: true, }, { Queries: []string{ "CREATE VIEW new_view AS SELECT 1;", "DROP VIEW new_view;", }, ExpectingErr: true, }, { Queries: []string{ "GRANT DROP ON *.* TO tester@localhost", "CREATE DATABASE new_db5;", "GRANT DROP ON *.* TO tester@localhost", "DROP DATABASE new_db5;", }, }, { Queries: []string{ "GRANT DROP ON *.* TO tester@localhost", "CREATE TABLE mydb.new_table (pk BIGINT PRIMARY KEY);", "DROP TABLE mydb.new_table;", }, }, { Queries: []string{ "GRANT DROP ON *.* TO tester@localhost", "CREATE TABLE mydb.new_table1 (pk BIGINT PRIMARY KEY);", "CREATE TABLE mydb.new_table2 (pk BIGINT PRIMARY KEY);", "DROP TABLE mydb.new_table1, mydb.new_table2;", }, }, { Queries: []string{ "GRANT DROP ON *.* TO tester@localhost", "CREATE VIEW new_view AS SELECT 1;", "DROP VIEW new_view;", }, }, { Queries: []string{ "GRANT DROP ON mydb.* TO tester@localhost", "CREATE TABLE mydb.new_table (pk BIGINT PRIMARY KEY);", "DROP TABLE mydb.new_table;", }, }, { Queries: []string{ "GRANT DROP ON mydb.* TO tester@localhost", "CREATE TABLE mydb.new_table1 (pk BIGINT PRIMARY KEY);", "CREATE TABLE mydb.new_table2 (pk BIGINT PRIMARY KEY);", "DROP TABLE mydb.new_table1, mydb.new_table2;", }, }, { Queries: []string{ "GRANT DROP ON mydb.new_table TO tester@localhost", "CREATE TABLE mydb.new_table (pk BIGINT PRIMARY KEY);", "DROP TABLE mydb.new_table;", }, }, { Queries: []string{ "GRANT DROP ON mydb.new_table1 TO tester@localhost", "CREATE TABLE mydb.new_table1 (pk BIGINT PRIMARY KEY);", "CREATE TABLE mydb.new_table2 (pk BIGINT PRIMARY KEY);", "DROP TABLE mydb.new_table1, mydb.new_table2;", }, ExpectingErr: true, }, { Queries: []string{ "GRANT DROP ON mydb.new_table2 TO tester@localhost", "CREATE TABLE mydb.new_table1 (pk BIGINT PRIMARY KEY);", "CREATE TABLE mydb.new_table2 (pk BIGINT PRIMARY KEY);", "DROP TABLE mydb.new_table1, mydb.new_table2;", }, ExpectingErr: true, }, { Queries: []string{ "GRANT DROP ON mydb.new_table1 TO tester@localhost", "GRANT DROP ON mydb.new_table2 TO tester@localhost", "CREATE TABLE mydb.new_table1 (pk BIGINT PRIMARY KEY);", "CREATE TABLE mydb.new_table2 (pk BIGINT PRIMARY KEY);", "DROP TABLE mydb.new_table1, mydb.new_table2;", }, }, { Queries: []string{ "CREATE ROLE new_role;", "DROP ROLE new_role;", }, ExpectingErr: true, }, { Queries: []string{ "GRANT DROP ROLE ON *.* TO tester@localhost", "CREATE ROLE new_role;", "DROP ROLE new_role;", }, }, { Queries: []string{ "CREATE INDEX new_idx ON mydb.test (v1);", }, ExpectingErr: true, }, { Queries: []string{ "CREATE INDEX new_idx ON mydb.test (v1);", "DROP INDEX new_idx ON mydb.test;", }, ExpectingErr: true, }, { Queries: []string{ "GRANT INDEX ON *.* TO tester@localhost", "CREATE INDEX new_idx ON mydb.test (v1);", }, }, { Queries: []string{ "GRANT INDEX ON *.* TO tester@localhost", "CREATE INDEX new_idx ON mydb.test (v1);", "DROP INDEX new_idx ON mydb.test;", }, }, { Queries: []string{ "GRANT INDEX ON mydb.* TO tester@localhost", "CREATE INDEX new_idx ON mydb.test (v1);", }, }, { Queries: []string{ "GRANT INDEX ON mydb.* TO tester@localhost", "CREATE INDEX new_idx ON mydb.test (v1);", "DROP INDEX new_idx ON mydb.test;", }, }, { Queries: []string{ "GRANT INDEX ON mydb.test TO tester@localhost", "CREATE INDEX new_idx ON mydb.test (v1);", }, }, { Queries: []string{ "GRANT INDEX ON mydb.test TO tester@localhost", "CREATE INDEX new_idx ON mydb.test (v1);", "DROP INDEX new_idx ON mydb.test;", }, }, { Queries: []string{ "INSERT INTO mydb.test VALUES (9, 9);", }, ExpectingErr: true, }, { Queries: []string{ "GRANT INSERT ON *.* TO tester@localhost", "INSERT INTO mydb.test VALUES (9, 9);", }, }, { Queries: []string{ "GRANT INSERT ON mydb.* TO tester@localhost", "INSERT INTO mydb.test VALUES (9, 9);", }, }, { Queries: []string{ "GRANT INSERT ON mydb.test TO tester@localhost", "INSERT INTO mydb.test VALUES (9, 9);", }, }, { Queries: []string{ "CREATE TRIGGER new_trig BEFORE INSERT ON mydb.test2 FOR EACH ROW SET NEW.v1 = NEW.pk * NEW.v1;", }, ExpectingErr: true, }, { Queries: []string{ "CREATE TRIGGER new_trig BEFORE INSERT ON mydb.test2 FOR EACH ROW SET NEW.v1 = NEW.pk * NEW.v1;", "DROP TRIGGER new_trig;", }, ExpectingErr: true, }, { Queries: []string{ "GRANT TRIGGER ON *.* TO tester@localhost", "CREATE TRIGGER new_trig BEFORE INSERT ON mydb.test2 FOR EACH ROW SET NEW.v1 = NEW.pk * NEW.v1;", }, }, { Queries: []string{ "GRANT TRIGGER ON *.* TO tester@localhost", "CREATE TRIGGER new_trig BEFORE INSERT ON mydb.test2 FOR EACH ROW SET NEW.v1 = NEW.pk * NEW.v1;", "DROP TRIGGER new_trig;", }, }, { Queries: []string{ "GRANT TRIGGER ON mydb.* TO tester@localhost", "CREATE TRIGGER new_trig BEFORE INSERT ON mydb.test2 FOR EACH ROW SET NEW.v1 = NEW.pk * NEW.v1;", }, }, { Queries: []string{ "GRANT TRIGGER ON mydb.* TO tester@localhost", "CREATE TRIGGER new_trig BEFORE INSERT ON mydb.test2 FOR EACH ROW SET NEW.v1 = NEW.pk * NEW.v1;", "DROP TRIGGER new_trig;", }, }, { Queries: []string{ "UPDATE mydb.test SET v1 = 0;", }, ExpectingErr: true, }, { Queries: []string{ "GRANT UPDATE ON *.* TO tester@localhost", "UPDATE mydb.test SET v1 = 0;", }, }, { Queries: []string{ "GRANT UPDATE ON mydb.* TO tester@localhost", "UPDATE mydb.test SET v1 = 0;", }, }, { Queries: []string{ "GRANT UPDATE ON mydb.test TO tester@localhost", "UPDATE mydb.test SET v1 = 0;", }, }, { Queries: []string{ "FLUSH PRIVILEGES;", }, ExpectingErr: true, }, { Queries: []string{ "GRANT RELOAD ON *.* TO tester@localhost", "FLUSH PRIVILEGES;", }, }, }
QuickPrivTests are test that specifically attempt to test as many privileges against as many statements as possible, while being as succinct as possible. All tests here could be fully represented as a UserPrivilegeTest, however each equivalent test would comparatively take up many more lines. This is intended to have as many tests as possible that are as quick to write as possible.
var ReplaceErrorTests = []GenericErrorQueryTest{
{
Name: "too few values",
Query: "REPLACE INTO mytable (s, i) VALUES ('x');",
},
{
Name: "too many values one column",
Query: "REPLACE INTO mytable (s) VALUES ('x', 999);",
},
{
Name: "too many values two columns",
Query: "REPLACE INTO mytable (i, s) VALUES (999, 'x', 'y');",
},
{
Name: "too few values no columns specified",
Query: "REPLACE INTO mytable VALUES (999);",
},
{
Name: "too many values no columns specified",
Query: "REPLACE INTO mytable VALUES (999, 'x', 'y');",
},
{
Name: "non-existent column values",
Query: "REPLACE INTO mytable (i, s, z) VALUES (999, 'x', 999);",
},
{
Name: "non-existent column set",
Query: "REPLACE INTO mytable SET i = 999, s = 'x', z = 999;",
},
{
Name: "duplicate column values",
Query: "REPLACE INTO mytable (i, s, s) VALUES (999, 'x', 'x');",
},
{
Name: "duplicate column set",
Query: "REPLACE INTO mytable SET i = 999, s = 'y', s = 'y';",
},
{
Name: "null given to non-nullable values",
Query: "INSERT INTO mytable (i, s) VALUES (null, 'y');",
},
{
Name: "null given to non-nullable set",
Query: "INSERT INTO mytable SET i = null, s = 'y';",
},
}
var ReplaceQueries = []WriteQueryTest{ { WriteQuery: "REPLACE INTO mytable VALUES (1, 'first row');", ExpectedWriteResult: []sql.Row{{sql.NewOkResult(2)}}, SelectQuery: "SELECT s FROM mytable WHERE i = 1;", ExpectedSelect: []sql.Row{{"first row"}}, }, { WriteQuery: "REPLACE INTO mytable SET i = 1, s = 'first row';", ExpectedWriteResult: []sql.Row{{sql.NewOkResult(2)}}, SelectQuery: "SELECT s FROM mytable WHERE i = 1;", ExpectedSelect: []sql.Row{{"first row"}}, }, { WriteQuery: "REPLACE INTO mytable VALUES (1, 'new row same i');", ExpectedWriteResult: []sql.Row{{sql.NewOkResult(2)}}, SelectQuery: "SELECT s FROM mytable WHERE i = 1;", ExpectedSelect: []sql.Row{{"new row same i"}}, }, { WriteQuery: "REPLACE INTO mytable SET i = 1, s = 'new row same i';", ExpectedWriteResult: []sql.Row{{sql.NewOkResult(2)}}, SelectQuery: "SELECT s FROM mytable WHERE i = 1;", ExpectedSelect: []sql.Row{{"new row same i"}}, }, { WriteQuery: "REPLACE INTO mytable (s, i) VALUES ('x', 999);", ExpectedWriteResult: []sql.Row{{sql.NewOkResult(1)}}, SelectQuery: "SELECT i FROM mytable WHERE s = 'x';", ExpectedSelect: []sql.Row{{int64(999)}}, }, { WriteQuery: "REPLACE INTO mytable SET s = 'x', i = 999;", ExpectedWriteResult: []sql.Row{{sql.NewOkResult(1)}}, SelectQuery: "SELECT i FROM mytable WHERE s = 'x';", ExpectedSelect: []sql.Row{{int64(999)}}, }, { WriteQuery: "REPLACE INTO mytable VALUES (999, 'x');", ExpectedWriteResult: []sql.Row{{sql.NewOkResult(1)}}, SelectQuery: "SELECT i FROM mytable WHERE s = 'x';", ExpectedSelect: []sql.Row{{int64(999)}}, }, { WriteQuery: "REPLACE INTO mytable SET i = 999, s = 'x';", ExpectedWriteResult: []sql.Row{{sql.NewOkResult(1)}}, SelectQuery: "SELECT i FROM mytable WHERE s = 'x';", ExpectedSelect: []sql.Row{{int64(999)}}, }, { WriteQuery: `REPLACE INTO typestable VALUES ( 999, 127, 32767, 2147483647, 9223372036854775807, 255, 65535, 4294967295, 18446744073709551615, 3.40282346638528859811704183484516925440e+38, 1.797693134862315708145274237317043567981e+308, '2037-04-05 12:51:36', '2231-11-07', 'random text', true, '{"key":"value"}', 'blobdata' );`, ExpectedWriteResult: []sql.Row{{sql.NewOkResult(1)}}, SelectQuery: "SELECT * FROM typestable WHERE id = 999;", ExpectedSelect: []sql.Row{{ int64(999), int8(math.MaxInt8), int16(math.MaxInt16), int32(math.MaxInt32), int64(math.MaxInt64), uint8(math.MaxUint8), uint16(math.MaxUint16), uint32(math.MaxUint32), uint64(math.MaxUint64), float32(math.MaxFloat32), float64(math.MaxFloat64), sql.MustConvert(sql.Timestamp.Convert("2037-04-05 12:51:36")), sql.MustConvert(sql.Date.Convert("2231-11-07")), "random text", sql.True, sql.MustJSON(`{"key":"value"}`), "blobdata", }}, }, { WriteQuery: `REPLACE INTO typestable SET id = 999, i8 = 127, i16 = 32767, i32 = 2147483647, i64 = 9223372036854775807, u8 = 255, u16 = 65535, u32 = 4294967295, u64 = 18446744073709551615, f32 = 3.40282346638528859811704183484516925440e+38, f64 = 1.797693134862315708145274237317043567981e+308, ti = '2037-04-05 12:51:36', da = '2231-11-07', te = 'random text', bo = true, js = '{"key":"value"}', bl = 'blobdata' ;`, ExpectedWriteResult: []sql.Row{{sql.NewOkResult(1)}}, SelectQuery: "SELECT * FROM typestable WHERE id = 999;", ExpectedSelect: []sql.Row{{ int64(999), int8(math.MaxInt8), int16(math.MaxInt16), int32(math.MaxInt32), int64(math.MaxInt64), uint8(math.MaxUint8), uint16(math.MaxUint16), uint32(math.MaxUint32), uint64(math.MaxUint64), float32(math.MaxFloat32), float64(math.MaxFloat64), sql.MustConvert(sql.Timestamp.Convert("2037-04-05 12:51:36")), sql.MustConvert(sql.Date.Convert("2231-11-07")), "random text", sql.True, sql.MustJSON(`{"key":"value"}`), "blobdata", }}, }, { WriteQuery: `REPLACE INTO typestable VALUES ( 999, -128, -32768, -2147483648, -9223372036854775808, 0, 0, 0, 0, 1.401298464324817070923729583289916131280e-45, 4.940656458412465441765687928682213723651e-324, '0000-00-00 00:00:00', '0000-00-00', '', false, '""', '' );`, ExpectedWriteResult: []sql.Row{{sql.NewOkResult(1)}}, SelectQuery: "SELECT * FROM typestable WHERE id = 999;", ExpectedSelect: []sql.Row{{ int64(999), int8(-math.MaxInt8 - 1), int16(-math.MaxInt16 - 1), int32(-math.MaxInt32 - 1), int64(-math.MaxInt64 - 1), uint8(0), uint16(0), uint32(0), uint64(0), float32(math.SmallestNonzeroFloat32), float64(math.SmallestNonzeroFloat64), sql.Timestamp.Zero(), sql.Date.Zero(), "", sql.False, sql.MustJSON(`""`), "", }}, }, { WriteQuery: `REPLACE INTO typestable SET id = 999, i8 = -128, i16 = -32768, i32 = -2147483648, i64 = -9223372036854775808, u8 = 0, u16 = 0, u32 = 0, u64 = 0, f32 = 1.401298464324817070923729583289916131280e-45, f64 = 4.940656458412465441765687928682213723651e-324, ti = '0000-00-00 00:00:00', da = '0000-00-00', te = '', bo = false, js = '""', bl = '' ;`, ExpectedWriteResult: []sql.Row{{sql.NewOkResult(1)}}, SelectQuery: "SELECT * FROM typestable WHERE id = 999;", ExpectedSelect: []sql.Row{{ int64(999), int8(-math.MaxInt8 - 1), int16(-math.MaxInt16 - 1), int32(-math.MaxInt32 - 1), int64(-math.MaxInt64 - 1), uint8(0), uint16(0), uint32(0), uint64(0), float32(math.SmallestNonzeroFloat32), float64(math.SmallestNonzeroFloat64), sql.Timestamp.Zero(), sql.Date.Zero(), "", sql.False, sql.MustJSON(`""`), "", }}, }, { WriteQuery: `REPLACE INTO typestable VALUES (999, null, null, null, null, null, null, null, null, null, null, null, null, null, null, null, null);`, ExpectedWriteResult: []sql.Row{{sql.NewOkResult(1)}}, SelectQuery: "SELECT * FROM typestable WHERE id = 999;", ExpectedSelect: []sql.Row{{int64(999), nil, nil, nil, nil, nil, nil, nil, nil, nil, nil, nil, nil, nil, nil, nil, nil}}, }, { WriteQuery: `REPLACE INTO typestable SET id=999, i8=null, i16=null, i32=null, i64=null, u8=null, u16=null, u32=null, u64=null, f32=null, f64=null, ti=null, da=null, te=null, bo=null, js=null, bl=null;`, ExpectedWriteResult: []sql.Row{{sql.NewOkResult(1)}}, SelectQuery: "SELECT * FROM typestable WHERE id = 999;", ExpectedSelect: []sql.Row{{int64(999), nil, nil, nil, nil, nil, nil, nil, nil, nil, nil, nil, nil, nil, nil, nil, nil}}, }, }
TODO: none of these tests insert into tables without primary key columns, which have different semantics for
REPLACE INTO queries. Add some tables / data without primary keys.
var RollbackTriggerTests = []ScriptTest{ { Name: "trigger before insert, reverts insert when query fails", SetUpScript: []string{ "create table a (i int primary key)", "create table b (x int)", "create trigger trig before insert on a for each row insert into b values (new.i);", }, Assertions: []ScriptTestAssertion{ { Query: "insert into a values (1), (2)", Expected: []sql.Row{ {sql.OkResult{RowsAffected: 2}}, }, }, { Query: "select x from b order by x", Expected: []sql.Row{ {1}, {2}, }, }, { Query: "insert into a values (1)", ExpectedErr: sql.ErrPrimaryKeyViolation, }, { Query: "select * from b", Expected: []sql.Row{ {1}, {2}, }, }, }, }, { Name: "trigger after insert, reverts insert when query fails", SetUpScript: []string{ "create table a (i int primary key)", "create table b (x int)", "create trigger trig after insert on a for each row insert into b values (new.i);", }, Assertions: []ScriptTestAssertion{ { Query: "insert into a values (1), (2)", Expected: []sql.Row{ {sql.OkResult{RowsAffected: 2}}, }, }, { Query: "select x from b order by x", Expected: []sql.Row{ {1}, {2}, }, }, { Query: "insert into a values (1)", ExpectedErr: sql.ErrPrimaryKeyViolation, }, { Query: "select * from b", Expected: []sql.Row{ {1}, {2}, }, }, }, }, { Name: "trigger before insert, reverts update when query fails", SetUpScript: []string{ "create table a (i int primary key)", "create table b (x int)", "insert into b values (0)", "create trigger trig before insert on a for each row update b set x = x + 1;", }, Assertions: []ScriptTestAssertion{ { Query: "insert into a values (1), (2)", Expected: []sql.Row{ {sql.OkResult{RowsAffected: 2}}, }, }, { Query: "select * from b", Expected: []sql.Row{ {2}, }, }, { Query: "insert into a values (1)", ExpectedErr: sql.ErrPrimaryKeyViolation, }, { Query: "select * from b", Expected: []sql.Row{ {2}, }, }, }, }, { Name: "trigger after insert, reverts update when query fails", SetUpScript: []string{ "create table a (i int primary key)", "create table b (x int)", "insert into b values (0)", "create trigger trig after insert on a for each row update b set x = x + 1;", }, Assertions: []ScriptTestAssertion{ { Query: "insert into a values (1), (2)", Expected: []sql.Row{ {sql.OkResult{RowsAffected: 2}}, }, }, { Query: "select * from b", Expected: []sql.Row{ {2}, }, }, { Query: "insert into a values (1)", ExpectedErr: sql.ErrPrimaryKeyViolation, }, { Query: "select * from b", Expected: []sql.Row{ {2}, }, }, }, }, { Name: "trigger before insert, reverts delete when query fails", SetUpScript: []string{ "create table a (i int primary key)", "create table b (x int)", "insert into a values (1)", "insert into b values (1), (2)", "create trigger trig before insert on a for each row delete from b where x = new.i;", }, Assertions: []ScriptTestAssertion{ { Query: "insert into a values (2)", Expected: []sql.Row{ {sql.OkResult{RowsAffected: 1}}, }, }, { Query: "select x from b order by x", Expected: []sql.Row{ {1}, }, }, { Query: "insert into a values (1)", ExpectedErr: sql.ErrPrimaryKeyViolation, }, { Query: "select * from b", Expected: []sql.Row{ {1}, }, }, }, }, { Name: "trigger after insert, reverts delete when query fails", SetUpScript: []string{ "create table a (i int primary key)", "create table b (x int)", "insert into a values (1)", "insert into b values (1), (2)", "create trigger trig after insert on a for each row delete from b where x = new.i;", }, Assertions: []ScriptTestAssertion{ { Query: "insert into a values (2)", Expected: []sql.Row{ {sql.OkResult{RowsAffected: 1}}, }, }, { Query: "select x from b order by x", Expected: []sql.Row{ {1}, }, }, { Query: "insert into a values (1)", ExpectedErr: sql.ErrPrimaryKeyViolation, }, { Query: "select * from b", Expected: []sql.Row{ {1}, }, }, }, }, { Name: "trigger before insert, reverts multiple inserts when query fails", SetUpScript: []string{ "create table a (i int primary key)", "create table b (x int)", "create trigger trig before insert on a for each row insert into b values (new.i);", }, Assertions: []ScriptTestAssertion{ { Query: "insert into a values (1), (1)", ExpectedErr: sql.ErrPrimaryKeyViolation, }, { Query: "select * from a", Expected: []sql.Row{}, }, { Query: "select * from b", Expected: []sql.Row{}, }, { Query: "insert into a values (0)", Expected: []sql.Row{ {sql.OkResult{RowsAffected: 1}}, }, }, { Query: "insert into a values (1), (2), (0)", ExpectedErr: sql.ErrPrimaryKeyViolation, }, { Query: "select * from a", Expected: []sql.Row{ {0}, }, }, { Query: "select * from b", Expected: []sql.Row{ {0}, }, }, }, }, { Name: "trigger after insert, reverts multiple inserts when query fails", SetUpScript: []string{ "create table a (i int primary key)", "create table b (x int)", "create trigger trig after insert on a for each row insert into b values (new.i);", }, Assertions: []ScriptTestAssertion{ { Query: "insert into a values (1), (1)", ExpectedErr: sql.ErrPrimaryKeyViolation, }, { Query: "select * from a", Expected: []sql.Row{}, }, { Query: "select * from b", Expected: []sql.Row{}, }, { Query: "insert into a values (0)", Expected: []sql.Row{ {sql.OkResult{RowsAffected: 1}}, }, }, { Query: "insert into a values (1), (2), (0)", ExpectedErr: sql.ErrPrimaryKeyViolation, }, { Query: "select * from a", Expected: []sql.Row{ {0}, }, }, { Query: "select * from b", Expected: []sql.Row{ {0}, }, }, }, }, { Name: "trigger before update, reverts insert when query fails", SetUpScript: []string{ "create table a (i int primary key)", "create table b (x int)", "insert into a values (0)", "create trigger trig before update on a for each row insert into b values (new.i);", }, Assertions: []ScriptTestAssertion{ { Query: "update a set i = 1", Expected: []sql.Row{ {sql.OkResult{ RowsAffected: 1, Info: plan.UpdateInfo{ Matched: 1, Updated: 1, }, }}, }, }, { Query: "select x from b", Expected: []sql.Row{ {1}, }, }, { Query: "update a set i = 'not int'", ExpectedErrStr: "error: 'not int' is not a valid value for 'INT'", }, { Query: "select * from b", Expected: []sql.Row{ {1}, }, }, }, }, { Name: "trigger after update, reverts insert when query fails", SetUpScript: []string{ "create table a (i int primary key)", "create table b (x int)", "insert into a values (0)", "create trigger trig after update on a for each row insert into b values (new.i);", }, Assertions: []ScriptTestAssertion{ { Query: "update a set i = 1", Expected: []sql.Row{ {sql.OkResult{ RowsAffected: 1, Info: plan.UpdateInfo{ Matched: 1, Updated: 1, }, }}, }, }, { Query: "select x from b", Expected: []sql.Row{ {1}, }, }, { Query: "update a set i = 'not int'", ExpectedErrStr: "error: 'not int' is not a valid value for 'INT'", }, { Query: "select * from b", Expected: []sql.Row{ {1}, }, }, }, }, { Name: "trigger before update, reverts update when query fails", SetUpScript: []string{ "create table a (i int primary key)", "create table b (x int)", "insert into a values (0)", "insert into b values (0)", "create trigger trig before update on a for each row update b set x = x + new.i;", }, Assertions: []ScriptTestAssertion{ { Query: "update a set i = 1", Expected: []sql.Row{ {sql.OkResult{ RowsAffected: 1, Info: plan.UpdateInfo{ Matched: 1, Updated: 1, }, }}, }, }, { Query: "select x from b", Expected: []sql.Row{ {1}, }, }, { Query: "update a set i = 'not int'", ExpectedErrStr: "error: 'not int' is not a valid value for 'INT'", }, { Query: "select * from b", Expected: []sql.Row{ {1}, }, }, }, }, { Name: "trigger after update, reverts update when query fails", SetUpScript: []string{ "create table a (i int primary key)", "create table b (x int)", "insert into a values (0)", "insert into b values (0)", "create trigger trig after update on a for each row update b set x = x + new.i;", }, Assertions: []ScriptTestAssertion{ { Query: "update a set i = 1", Expected: []sql.Row{ {sql.OkResult{ RowsAffected: 1, Info: plan.UpdateInfo{ Matched: 1, Updated: 1, }, }}, }, }, { Query: "select x from b", Expected: []sql.Row{ {1}, }, }, { Query: "update a set i = 'not int'", ExpectedErrStr: "error: 'not int' is not a valid value for 'INT'", }, { Query: "select * from b", Expected: []sql.Row{ {1}, }, }, }, }, { Name: "trigger before update, reverts delete when query fails", SetUpScript: []string{ "create table a (i int primary key)", "create table b (x int)", "insert into a values (0)", "insert into b values (1), (2)", "create trigger trig before update on a for each row delete from b where x = new.i;", }, Assertions: []ScriptTestAssertion{ { Query: "update a set i = 1", Expected: []sql.Row{ {sql.OkResult{ RowsAffected: 1, Info: plan.UpdateInfo{ Matched: 1, Updated: 1, }, }}, }, }, { Query: "select x from b", Expected: []sql.Row{ {2}, }, }, { Query: "update a set i = 'not int'", ExpectedErrStr: "error: 'not int' is not a valid value for 'INT'", }, { Query: "select * from b", Expected: []sql.Row{ {2}, }, }, }, }, { Name: "trigger after update, reverts delete when query fails", SetUpScript: []string{ "create table a (i int primary key)", "create table b (x int)", "insert into a values (0)", "insert into b values (1), (2)", "create trigger trig after update on a for each row delete from b where x = new.i;", }, Assertions: []ScriptTestAssertion{ { Query: "update a set i = 1", Expected: []sql.Row{ {sql.OkResult{ RowsAffected: 1, Info: plan.UpdateInfo{ Matched: 1, Updated: 1, }, }}, }, }, { Query: "select x from b", Expected: []sql.Row{ {2}, }, }, { Query: "update a set i = 'not int'", ExpectedErrStr: "error: 'not int' is not a valid value for 'INT'", }, { Query: "select * from b", Expected: []sql.Row{ {2}, }, }, }, }, { Name: "triggers before and after insert fails, rollback", SetUpScript: []string{ "create table a (x int primary key)", "create table b (y int primary key)", "create trigger a1 before insert on a for each row insert into b values (NEW.x * 7)", "create trigger a2 after insert on a for each row insert into b values (New.x * 11)", }, Assertions: []ScriptTestAssertion{ { Query: "insert into a values (2), (3), (5)", Expected: []sql.Row{ {sql.NewOkResult(3)}, }, }, { Query: "select x from a order by 1", Expected: []sql.Row{ {2}, {3}, {5}, }, }, { Query: "select y from b order by 1", Expected: []sql.Row{ {14}, {21}, {22}, {33}, {35}, {55}, }, }, { Query: "insert into a values (2), (3), (5)", ExpectedErr: sql.ErrPrimaryKeyViolation, }, { Query: "select x from a order by 1", Expected: []sql.Row{ {2}, {3}, {5}, }, }, { Query: "select y from b order by 1", Expected: []sql.Row{ {14}, {21}, {22}, {33}, {35}, {55}, }, }, }, }, { Name: "autocommit off, trigger before insert, reverts insert when query fails", SetUpScript: []string{ "set @@autocommit = off", "create table a (i int primary key)", "create table b (x int)", "create trigger trig before insert on a for each row insert into b values (new.i);", }, Assertions: []ScriptTestAssertion{ { Query: "insert into a values (1), (2)", Expected: []sql.Row{ {sql.OkResult{RowsAffected: 2}}, }, }, { Query: "select x from b order by x", Expected: []sql.Row{ {1}, {2}, }, }, { Query: "insert into a values (1)", ExpectedErr: sql.ErrPrimaryKeyViolation, }, { Query: "select * from b", Expected: []sql.Row{ {1}, {2}, }, }, }, }, { Name: "trigger before update, reverts insert when query fails", SetUpScript: []string{ "set @@autocommit = off", "create table a (i int primary key)", "create table b (x int)", "insert into a values (0)", "create trigger trig before update on a for each row insert into b values (new.i);", }, Assertions: []ScriptTestAssertion{ { Query: "update a set i = 1", Expected: []sql.Row{ {sql.OkResult{ RowsAffected: 1, Info: plan.UpdateInfo{ Matched: 1, Updated: 1, }, }}, }, }, { Query: "select x from b", Expected: []sql.Row{ {1}, }, }, { Query: "update a set i = 'not int'", ExpectedErrStr: "error: 'not int' is not a valid value for 'INT'", }, { Query: "select * from b", Expected: []sql.Row{ {1}, }, }, }, }, }
RollbackTriggerTests are trigger tests that require rollback logic to work correctly
var ScriptTests = []ScriptTest{ { Name: "failed statements data validation for INSERT, UPDATE", SetUpScript: []string{ "CREATE TABLE test (pk BIGINT PRIMARY KEY, v1 BIGINT, INDEX (v1));", "INSERT INTO test VALUES (1,1), (4,4), (5,5);", }, Assertions: []ScriptTestAssertion{ { Query: "INSERT INTO test VALUES (2,2), (3,3), (1,1);", ExpectedErrStr: "duplicate primary key given: [1]", }, { Query: "SELECT * FROM test;", Expected: []sql.Row{{1, 1}, {4, 4}, {5, 5}}, }, { Query: "UPDATE test SET pk = pk + 1 ORDER BY pk;", ExpectedErrStr: "duplicate primary key given: [5]", }, { Query: "SELECT * FROM test;", Expected: []sql.Row{{1, 1}, {4, 4}, {5, 5}}, }, }, }, { Name: "failed statements data validation for DELETE, REPLACE", SetUpScript: []string{ "CREATE TABLE test (pk BIGINT PRIMARY KEY, v1 BIGINT, INDEX (v1));", "INSERT INTO test VALUES (1,1), (4,4), (5,5);", "CREATE TABLE test2 (pk BIGINT PRIMARY KEY, CONSTRAINT fk_test FOREIGN KEY (pk) REFERENCES test (v1));", "INSERT INTO test2 VALUES (4);", }, Assertions: []ScriptTestAssertion{ { Query: "DELETE FROM test WHERE pk > 0;", ExpectedErr: sql.ErrForeignKeyParentViolation, }, { Query: "SELECT * FROM test;", Expected: []sql.Row{{1, 1}, {4, 4}, {5, 5}}, }, { Query: "SELECT * FROM test2;", Expected: []sql.Row{{4}}, }, { Query: "REPLACE INTO test VALUES (1,7), (4,8), (5,9);", ExpectedErr: sql.ErrForeignKeyParentViolation, }, { Query: "SELECT * FROM test;", Expected: []sql.Row{{1, 1}, {4, 4}, {5, 5}}, }, { Query: "SELECT * FROM test2;", Expected: []sql.Row{{4}}, }, }, }, { Name: "delete with in clause", SetUpScript: []string{ "create table a (x int primary key)", "insert into a values (1), (3), (5)", "delete from a where x in (1, 3)", }, Query: "select x from a order by 1", Expected: []sql.Row{ {5}, }, }, { Name: "sqllogictest evidence/slt_lang_aggfunc.test", SetUpScript: []string{ "CREATE TABLE t1( x INTEGER, y VARCHAR(8) )", "INSERT INTO t1 VALUES(1,'true')", "INSERT INTO t1 VALUES(0,'false')", "INSERT INTO t1 VALUES(NULL,'NULL')", }, Query: "SELECT count(DISTINCT x) FROM t1", Expected: []sql.Row{ {2}, }, }, { Name: "sqllogictest index/commute/10/slt_good_1.test", SetUpScript: []string{ "CREATE TABLE tab0(pk INTEGER PRIMARY KEY, col0 INTEGER, col1 FLOAT, col2 TEXT, col3 INTEGER, col4 FLOAT, col5 TEXT)", "INSERT INTO tab0 VALUES(0,42,58.92,'fnbtk',54,68.41,'xmttf')", "INSERT INTO tab0 VALUES(1,31,46.55,'sksjf',46,53.20,'wiuva')", "INSERT INTO tab0 VALUES(2,30,31.11,'oldqn',41,5.26,'ulaay')", "INSERT INTO tab0 VALUES(3,77,44.90,'pmsir',70,84.14,'vcmyo')", "INSERT INTO tab0 VALUES(4,23,95.26,'qcwxh',32,48.53,'rvtbr')", "INSERT INTO tab0 VALUES(5,43,6.75,'snvwg',3,14.38,'gnfxz')", "INSERT INTO tab0 VALUES(6,47,98.26,'bzzva',60,15.2,'imzeq')", "INSERT INTO tab0 VALUES(7,98,40.9,'lsrpi',78,66.30,'ephwy')", "INSERT INTO tab0 VALUES(8,19,15.16,'ycvjz',55,38.70,'dnkkz')", "INSERT INTO tab0 VALUES(9,7,84.4,'ptovf',17,2.46,'hrxsf')", "CREATE TABLE tab1(pk INTEGER PRIMARY KEY, col0 INTEGER, col1 FLOAT, col2 TEXT, col3 INTEGER, col4 FLOAT, col5 TEXT)", "CREATE INDEX idx_tab1_0 on tab1 (col0)", "CREATE INDEX idx_tab1_1 on tab1 (col1)", "CREATE INDEX idx_tab1_3 on tab1 (col3)", "CREATE INDEX idx_tab1_4 on tab1 (col4)", "INSERT INTO tab1 SELECT * FROM tab0", "CREATE TABLE tab2(pk INTEGER PRIMARY KEY, col0 INTEGER, col1 FLOAT, col2 TEXT, col3 INTEGER, col4 FLOAT, col5 TEXT)", "CREATE UNIQUE INDEX idx_tab2_1 ON tab2 (col4 DESC,col3)", "CREATE UNIQUE INDEX idx_tab2_2 ON tab2 (col3 DESC,col0)", "CREATE UNIQUE INDEX idx_tab2_3 ON tab2 (col3 DESC,col1)", "INSERT INTO tab2 SELECT * FROM tab0", "CREATE TABLE tab3(pk INTEGER PRIMARY KEY, col0 INTEGER, col1 FLOAT, col2 TEXT, col3 INTEGER, col4 FLOAT, col5 TEXT)", "CREATE INDEX idx_tab3_0 ON tab3 (col3 DESC)", "INSERT INTO tab3 SELECT * FROM tab0", "CREATE TABLE tab4(pk INTEGER PRIMARY KEY, col0 INTEGER, col1 FLOAT, col2 TEXT, col3 INTEGER, col4 FLOAT, col5 TEXT)", "CREATE INDEX idx_tab4_0 ON tab4 (col0 DESC)", "CREATE UNIQUE INDEX idx_tab4_2 ON tab4 (col4 DESC,col3)", "CREATE INDEX idx_tab4_3 ON tab4 (col3 DESC)", "INSERT INTO tab4 SELECT * FROM tab0", }, Query: "SELECT pk FROM tab2 WHERE 78 < col0 AND 19 < col3", Expected: []sql.Row{ {7}, }, }, { Name: "3 tables, linear join", SetUpScript: []string{ "create table a (xa int primary key, ya int, za int)", "create table b (xb int primary key, yb int, zb int)", "create table c (xc int primary key, yc int, zc int)", "insert into a values (1,2,3)", "insert into b values (1,2,3)", "insert into c values (1,2,3)", }, Assertions: []ScriptTestAssertion{ { Query: "select ya from a join b on ya - 1= xb join c on xc = zb - 2", Expected: []sql.Row{{2}}, }, }, }, { Name: "3 tables, v join", SetUpScript: []string{ "create table a (xa int primary key, ya int, za int)", "create table b (xb int primary key, yb int, zb int)", "create table c (xc int primary key, yc int, zc int)", "insert into a values (1,2,3)", "insert into b values (1,2,3)", "insert into c values (1,2,3)", }, Assertions: []ScriptTestAssertion{ { Query: "select za from a join b on ya - 1 = xb join c on xa = xc", Expected: []sql.Row{{3}}, }, }, }, { Name: "3 tables, linear join, indexes on A,C", SetUpScript: []string{ "create table a (xa int primary key, ya int, za int)", "create table b (xb int primary key, yb int, zb int)", "create table c (xc int primary key, yc int, zc int)", "insert into a values (1,2,3)", "insert into b values (1,2,3)", "insert into c values (1,2,3)", }, Assertions: []ScriptTestAssertion{ { Query: "select xa from a join b on xa = yb - 1 join c on yb - 1 = xc", Expected: []sql.Row{{1}}, }, }, }, { Name: "4 tables, linear join", SetUpScript: []string{ "create table a (xa int primary key, ya int, za int)", "create table b (xb int primary key, yb int, zb int)", "create table c (xc int primary key, yc int, zc int)", "create table d (xd int primary key, yd int, zd int)", "insert into a values (1,2,3)", "insert into b values (1,2,3)", "insert into c values (1,2,3)", "insert into d values (1,2,3)", }, Assertions: []ScriptTestAssertion{ { Query: "select xa from a join b on ya - 1 = xb join c on xb = xc join d on xc = xd", Expected: []sql.Row{{1}}, }, }, }, { Name: "4 tables, linear join, index on D", SetUpScript: []string{ "create table a (xa int primary key, ya int, za int)", "create table b (xb int primary key, yb int, zb int)", "create table c (xc int primary key, yc int, zc int)", "create table d (xd int primary key, yd int, zd int)", "insert into a values (1,2,3)", "insert into b values (1,2,3)", "insert into c values (1,2,3)", "insert into d values (1,2,3)", }, Assertions: []ScriptTestAssertion{ { Query: "select xa from a join b on ya = yb join c on yb = yc join d on yc - 1 = xd", Expected: []sql.Row{{1}}, }, }, }, { Name: "4 tables, left join, indexes on all tables", SetUpScript: []string{ "create table a (xa int primary key, ya int, za int)", "create table b (xb int primary key, yb int, zb int)", "create table c (xc int primary key, yc int, zc int)", "create table d (xd int primary key, yd int, zd int)", "insert into a values (1,2,3)", "insert into b values (1,2,3)", "insert into c values (1,2,3)", "insert into d values (1,2,3)", }, Assertions: []ScriptTestAssertion{ { Query: "select xa from a left join b on ya = yb left join c on yb = yc left join d on yc - 1 = xd", Expected: []sql.Row{{1}}, }, }, }, { Name: "4 tables, linear join, index on B, D", SetUpScript: []string{ "create table a (xa int primary key, ya int, za int)", "create table b (xb int primary key, yb int, zb int)", "create table c (xc int primary key, yc int, zc int)", "create table d (xd int primary key, yd int, zd int)", "insert into a values (1,2,3)", "insert into b values (1,2,3)", "insert into c values (1,2,3)", "insert into d values (1,2,3)", }, Assertions: []ScriptTestAssertion{ { Query: "select xa from a join b on ya - 1 = xb join c on yc = za - 1 join d on yc - 1 = xd", Expected: []sql.Row{{1}}, }, }, }, { Name: "4 tables, all joined to A", SetUpScript: []string{ "create table a (xa int primary key, ya int, za int)", "create table b (xb int primary key, yb int, zb int)", "create table c (xc int primary key, yc int, zc int)", "create table d (xd int primary key, yd int, zd int)", "insert into a values (1,2,3)", "insert into b values (1,2,3)", "insert into c values (1,2,3)", "insert into d values (1,2,3)", }, Assertions: []ScriptTestAssertion{ { Query: "select xa from a join b on xa = xb join c on ya - 1 = xc join d on za - 2 = xd", Expected: []sql.Row{{1}}, }, }, }, { Name: "4 tables, all joined to D", SetUpScript: []string{ "create table a (xa int primary key, ya int, za int)", "create table b (xb int primary key, yb int, zb int)", "create table c (xc int primary key, yc int, zc int)", "create table d (xd int primary key, yd int, zd int)", "insert into a values (1,2,3)", "insert into b values (1,2,3)", "insert into c values (1,2,3)", "insert into d values (1,2,3)", }, Assertions: []ScriptTestAssertion{ { Query: "select xa from d join a on yd - 1 = xa join c on zd - 2 = xc join b on xb = zd - 2", Expected: []sql.Row{{1}}, }, }, }, { Name: "5 tables, complex join conditions", SetUpScript: []string{ "create table a (xa int primary key, ya int, za int)", "create table b (xb int primary key, yb int, zb int)", "create table c (xc int primary key, yc int, zc int)", "create table d (xd int primary key, yd int, zd int)", "create table e (xe int, ye int, ze int, primary key(xe, ye))", "insert into a values (1,2,3)", "insert into b values (1,2,3)", "insert into c values (1,2,3)", "insert into d values (1,2,3)", "insert into e values (1,2,3)", }, Assertions: []ScriptTestAssertion{ { Query: `select xa from a join b on ya - 1 = xb join c on xc = za - 2 join d on xd = yb - 1 join e on xe = zb - 2 and ye = yc`, Expected: []sql.Row{{1}}, }, }, }, { Name: "UUIDs used in the wild.", SetUpScript: []string{ "SET @uuid = '6ccd780c-baba-1026-9564-5b8c656024db'", "SET @binuuid = '0011223344556677'", }, Assertions: []ScriptTestAssertion{ { Query: `SELECT IS_UUID(UUID())`, Expected: []sql.Row{{int8(1)}}, }, { Query: `SELECT IS_UUID(@uuid)`, Expected: []sql.Row{{int8(1)}}, }, { Query: `SELECT BIN_TO_UUID(UUID_TO_BIN(@uuid))`, Expected: []sql.Row{{"6ccd780c-baba-1026-9564-5b8c656024db"}}, }, { Query: `SELECT BIN_TO_UUID(UUID_TO_BIN(@uuid, 1), 1)`, Expected: []sql.Row{{"6ccd780c-baba-1026-9564-5b8c656024db"}}, }, { Query: `SELECT UUID_TO_BIN(NULL)`, Expected: []sql.Row{{nil}}, }, { Query: `SELECT HEX(UUID_TO_BIN(@uuid))`, Expected: []sql.Row{{"6CCD780CBABA102695645B8C656024DB"}}, }, { Query: `SELECT UUID_TO_BIN(123)`, ExpectedErr: sql.ErrUuidUnableToParse, }, { Query: `SELECT BIN_TO_UUID(123)`, ExpectedErr: sql.ErrUuidUnableToParse, }, { Query: `SELECT BIN_TO_UUID(X'00112233445566778899aabbccddeeff')`, Expected: []sql.Row{{"00112233-4455-6677-8899-aabbccddeeff"}}, }, { Query: `SELECT BIN_TO_UUID('0011223344556677')`, Expected: []sql.Row{{"30303131-3232-3333-3434-353536363737"}}, }, { Query: `SELECT BIN_TO_UUID(@binuuid)`, Expected: []sql.Row{{"30303131-3232-3333-3434-353536363737"}}, }, }, }, { Name: "Test cases on select into statement", SetUpScript: []string{ "SELECT 1 INTO @abc", "SELECT * FROM (VALUES ROW(22,44,88)) AS t INTO @x,@y,@z", "CREATE TABLE tab1 (id int primary key, v1 int)", "INSERT INTO tab1 VALUES (1, 1), (2, 3), (3, 6)", "SELECT id FROM tab1 ORDER BY id DESC LIMIT 1 INTO @myVar", "CREATE TABLE tab2 (i2 int primary key, s text)", "INSERT INTO tab2 VALUES (1, 'b'), (2, 'm'), (3, 'g')", "SELECT m.id, t.s FROM tab1 m JOIN tab2 t on m.id = t.i2 ORDER BY t.s DESC LIMIT 1 INTO @myId, @myText", "SELECT id FROM tab1 WHERE id > 3 UNION select s FROM tab2 WHERE s < 'f' INTO @mustSingleVar", }, Assertions: []ScriptTestAssertion{ { Query: `SELECT @abc`, Expected: []sql.Row{{int8(1)}}, }, { Query: `SELECT @z, @x, @y`, Expected: []sql.Row{{88, 22, 44}}, }, { Query: `SELECT @myVar, @mustSingleVar`, Expected: []sql.Row{{3, "b"}}, }, { Query: `SELECT @myId, @myText, @myUnion`, Expected: []sql.Row{{2, "m", nil}}, }, { Query: `SELECT id FROM tab1 ORDER BY id DESC INTO @myvar`, ExpectedErr: sql.ErrMoreThanOneRow, }, { Query: `SELECT 1 INTO OUTFILE 'x.txt'`, ExpectedErr: sql.ErrUnsupportedSyntax, }, { Query: `SELECT id INTO DUMPFILE 'dump.txt' FROM tab1 ORDER BY id DESC LIMIT 15`, ExpectedErr: sql.ErrUnsupportedSyntax, }, { Query: `select 1, 2, 3 into @my1, @my2`, ExpectedErr: sql.ErrColumnNumberDoesNotMatch, }, { Query: `SELECT id, v1 INTO @myFirstVar FROM tab1 ORDER BY id DESC LIMIT 1 INTO @mySecondVar`, ExpectedErrStr: "Multiple INTO clauses in one query block at position 84 near '@mySecondVar'", }, { Query: `SELECT id FROM tab1 WHERE id > 3 UNION select s INTO @mustSingleVar FROM tab2 WHERE s < 'f' ORDER BY s DESC`, ExpectedErrStr: "INTO clause is not allowed at position 98 near 'ORDER'", }, }, }, { Name: "CrossDB Queries", SetUpScript: []string{ "CREATE DATABASE test", "CREATE TABLE test.x (pk int primary key)", "insert into test.x values (1),(2),(3)", "DELETE FROM test.x WHERE pk=2", "UPDATE test.x set pk=300 where pk=3", "create table a (xa int primary key, ya int, za int)", "insert into a values (1,2,3)", }, Assertions: []ScriptTestAssertion{ { Query: "SELECT pk from test.x", Expected: []sql.Row{{1}, {300}}, }, { Query: "SELECT * from a", Expected: []sql.Row{{1, 2, 3}}, }, }, }, { Name: "Top-level DECLARE statements", Assertions: []ScriptTestAssertion{ { Query: "DECLARE no_such_table CONDITION FOR SQLSTATE '42S02'", ExpectedErr: sql.ErrSyntaxError, }, { Query: "DECLARE no_such_table CONDITION FOR 1051", ExpectedErr: sql.ErrSyntaxError, }, { Query: "DECLARE a CHAR(16)", ExpectedErr: sql.ErrSyntaxError, }, { Query: "DECLARE cur2 CURSOR FOR SELECT i FROM test.t2", ExpectedErr: sql.ErrSyntaxError, }, { Query: "DECLARE CONTINUE HANDLER FOR NOT FOUND SET done = TRUE", ExpectedErr: sql.ErrSyntaxError, }, }, }, { Name: "last_insert_id() behavior", SetUpScript: []string{ "create table a (x int primary key auto_increment, y int)", "create table b (x int primary key)", }, Assertions: []ScriptTestAssertion{ { Query: "select last_insert_id()", Expected: []sql.Row{{0}}, }, { Query: "insert into a (y) values (1)", Expected: []sql.Row{{sql.OkResult{RowsAffected: 1, InsertID: 1}}}, }, { Query: "select last_insert_id()", Expected: []sql.Row{{1}}, }, { Query: "insert into a (y) values (2), (3)", Expected: []sql.Row{{sql.OkResult{RowsAffected: 2, InsertID: 2}}}, }, { Query: "select last_insert_id()", Expected: []sql.Row{{2}}, }, { Query: "insert into b (x) values (1), (2)", Expected: []sql.Row{{sql.NewOkResult(2)}}, }, { Query: "select last_insert_id()", Expected: []sql.Row{{2}}, }, }, }, { Name: "row_count() behavior", SetUpScript: []string{ "create table b (x int primary key)", "insert into b values (1), (2), (3), (4)", }, Assertions: []ScriptTestAssertion{ { Query: "select row_count()", Expected: []sql.Row{{4}}, }, { Query: "replace into b values (1)", Expected: []sql.Row{{sql.NewOkResult(2)}}, }, { Query: "select row_count()", Expected: []sql.Row{{2}}, }, { Query: "select row_count()", Expected: []sql.Row{{-1}}, }, { Query: "select count(*) from b", Expected: []sql.Row{{4}}, }, { Query: "select row_count()", Expected: []sql.Row{{-1}}, }, { Query: "update b set x = x + 10 where x <> 2", Expected: []sql.Row{{sql.OkResult{ RowsAffected: 3, Info: plan.UpdateInfo{ Matched: 3, Updated: 3, }, }}}, }, { Query: "select row_count()", Expected: []sql.Row{{3}}, }, { Query: "select row_count()", Expected: []sql.Row{{-1}}, }, { Query: "delete from b where x <> 2", Expected: []sql.Row{{sql.NewOkResult(3)}}, }, { Query: "select row_count()", Expected: []sql.Row{{3}}, }, { Query: "select row_count()", Expected: []sql.Row{{-1}}, }, { Query: "alter table b add column y int null", Expected: []sql.Row{{sql.NewOkResult(0)}}, }, { Query: "select row_count()", Expected: []sql.Row{{0}}, }, { Query: "select row_count()", Expected: []sql.Row{{-1}}, }, }, }, { Name: "same alias names for result column name and alias table column name", SetUpScript: []string{ "CREATE TABLE tab0(col0 INTEGER, col1 INTEGER, col2 INTEGER)", "INSERT INTO tab0 VALUES(83,0,38)", "INSERT INTO tab0 VALUES(26,0,79)", "INSERT INTO tab0 VALUES(43,81,24)", }, Assertions: []ScriptTestAssertion{ { Query: "SELECT + 13 AS col0 FROM tab0 GROUP BY tab0.col0", Expected: []sql.Row{{13}, {13}, {13}}, }, { Query: "SELECT 82 col1 FROM tab0 AS cor0 GROUP BY cor0.col1", Expected: []sql.Row{{82}, {82}}, }, { Query: "SELECT - cor0.col2 * - col2 AS col1 FROM tab0 AS cor0 GROUP BY col2, cor0.col1", Expected: []sql.Row{{1444}, {6241}, {576}}, }, { Query: "SELECT ALL + 40 col1 FROM tab0 AS cor0 GROUP BY cor0.col1", Expected: []sql.Row{{40}, {40}}, }, { Query: "SELECT DISTINCT - cor0.col1 col1 FROM tab0 AS cor0 GROUP BY cor0.col1, cor0.col2", Expected: []sql.Row{{-81}, {0}}, }, { Query: "SELECT DISTINCT ( cor0.col0 ) - col0 AS col2 FROM tab0 AS cor0 GROUP BY cor0.col2, cor0.col0, cor0.col0", Expected: []sql.Row{{0}}, }, }, }, { Name: "found_rows() behavior", SetUpScript: []string{ "create table b (x int primary key)", "insert into b values (1), (2), (3), (4)", }, Assertions: []ScriptTestAssertion{ { Query: "select * from b where x < 2", Expected: []sql.Row{{1}}, }, { Query: "select found_rows()", Expected: []sql.Row{{1}}, }, { Query: "select * from b", Expected: []sql.Row{{1}, {2}, {3}, {4}}, }, { Query: "select found_rows()", Expected: []sql.Row{{4}}, }, { Query: "select found_rows()", Expected: []sql.Row{{1}}, }, { Query: "select * from b order by x limit 3", Expected: []sql.Row{{1}, {2}, {3}}, }, { Query: "select found_rows()", Expected: []sql.Row{{3}}, }, { Query: "select found_rows()", Expected: []sql.Row{{1}}, }, { Query: "select sql_calc_found_rows * from b order by x limit 3", Expected: []sql.Row{{1}, {2}, {3}}, }, { Query: "select found_rows()", Expected: []sql.Row{{4}}, }, { Query: "select found_rows()", Expected: []sql.Row{{1}}, }, { Query: "select sql_calc_found_rows * from b where x <= 2 order by x limit 1", Expected: []sql.Row{{1}}, }, { Query: "select found_rows()", Expected: []sql.Row{{2}}, }, { Query: "select sql_calc_found_rows * from b where x <= 2 order by x limit 1", Expected: []sql.Row{{1}}, }, { Query: "insert into b values (10), (11), (12), (13)", Expected: []sql.Row{{sql.NewOkResult(4)}}, }, { Query: "select found_rows()", Expected: []sql.Row{{2}}, }, { Query: "update b set x = x where x < 40", Expected: []sql.Row{{sql.OkResult{RowsAffected: 0, InsertID: 0, Info: plan.UpdateInfo{Matched: 8}}}}, }, { Query: "select found_rows()", Expected: []sql.Row{{8}}, }, { Query: "update b set x = x where x > 10", Expected: []sql.Row{{sql.OkResult{RowsAffected: 0, InsertID: 0, Info: plan.UpdateInfo{Matched: 3}}}}, }, { Query: "select found_rows()", Expected: []sql.Row{{3}}, }, }, }, { Name: "INSERT INTO ... SELECT with AUTO_INCREMENT", SetUpScript: []string{ "create table ai (pk int primary key auto_increment, c0 int);", "create table other (pk int primary key);", "insert into other values (1), (2), (3)", "insert into ai (c0) select * from other order by other.pk;", }, Query: "select * from ai;", Expected: []sql.Row{ {1, 1}, {2, 2}, {3, 3}, }, }, { Name: "Indexed Join On Keyless Table", SetUpScript: []string{ "create table l (pk int primary key, c0 int, c1 int);", "create table r (c0 int, c1 int, third int);", "create index r_c0 on r (c0);", "create index r_c1 on r (c1);", "create index r_third on r (third);", "insert into l values (0, 0, 0), (1, 0, 1), (2, 1, 0), (3, 0, 2), (4, 2, 0), (5, 1, 2), (6, 2, 1), (7, 2, 2);", "insert into l values (256, 1024, 4096);", "insert into r values (1, 1, -1), (2, 2, -1), (2, 2, -1);", "insert into r values (-1, -1, 256);", }, Assertions: []ScriptTestAssertion{ { Query: "select pk, l.c0, l.c1 from l join r on l.c0 = r.c0 or l.c1 = r.c1 order by 1, 2, 3;", Expected: []sql.Row{ {1, 0, 1}, {2, 1, 0}, {3, 0, 2}, {3, 0, 2}, {4, 2, 0}, {4, 2, 0}, {5, 1, 2}, {5, 1, 2}, {5, 1, 2}, {6, 2, 1}, {6, 2, 1}, {6, 2, 1}, {7, 2, 2}, {7, 2, 2}, }, }, { Query: "select pk, l.c0, l.c1 from l join r on l.c0 = r.c0 or l.c1 = r.c1 or l.pk = r.third order by 1, 2, 3;", Expected: []sql.Row{ {1, 0, 1}, {2, 1, 0}, {3, 0, 2}, {3, 0, 2}, {4, 2, 0}, {4, 2, 0}, {5, 1, 2}, {5, 1, 2}, {5, 1, 2}, {6, 2, 1}, {6, 2, 1}, {6, 2, 1}, {7, 2, 2}, {7, 2, 2}, {256, 1024, 4096}, }, }, { Query: "select pk, l.c0, l.c1 from l join r on l.c0 = r.c0 or l.c1 < 4 and l.c1 = r.c1 or l.c1 >= 4 and l.c1 = r.c1 order by 1, 2, 3;", Expected: []sql.Row{ {1, 0, 1}, {2, 1, 0}, {3, 0, 2}, {3, 0, 2}, {4, 2, 0}, {4, 2, 0}, {5, 1, 2}, {5, 1, 2}, {5, 1, 2}, {6, 2, 1}, {6, 2, 1}, {6, 2, 1}, {7, 2, 2}, {7, 2, 2}, }, }, }, }, { Name: "Group Concat Queries", SetUpScript: []string{ "CREATE TABLE x (pk int)", "INSERT INTO x VALUES (1),(2),(3),(4),(NULL)", "create table t (o_id int, attribute longtext, value longtext)", "INSERT INTO t VALUES (2, 'color', 'red'), (2, 'fabric', 'silk')", "INSERT INTO t VALUES (3, 'color', 'green'), (3, 'shape', 'square')", "create table nulls(pk int)", "INSERT INTO nulls VALUES (NULL)", }, Assertions: []ScriptTestAssertion{ { Query: `SELECT group_concat(pk ORDER BY pk) FROM x;`, Expected: []sql.Row{{"1,2,3,4"}}, }, { Query: `SELECT group_concat(DISTINCT pk ORDER BY pk) FROM x;`, Expected: []sql.Row{{"1,2,3,4"}}, }, { Query: `SELECT group_concat(DISTINCT pk ORDER BY pk SEPARATOR '-') FROM x;`, Expected: []sql.Row{{"1-2-3-4"}}, }, { Query: "SELECT group_concat(`attribute` ORDER BY `attribute`) FROM t group by o_id order by o_id asc", Expected: []sql.Row{{"color,fabric"}, {"color,shape"}}, }, { Query: "SELECT group_concat(DISTINCT `attribute` ORDER BY value DESC SEPARATOR ';') FROM t group by o_id order by o_id asc", Expected: []sql.Row{{"fabric;color"}, {"shape;color"}}, }, { Query: "SELECT group_concat(DISTINCT `attribute` ORDER BY `attribute`) FROM t", Expected: []sql.Row{{"color,fabric,shape"}}, }, { Query: "SELECT group_concat(`attribute` ORDER BY `attribute`) FROM t", Expected: []sql.Row{{"color,color,fabric,shape"}}, }, { Query: `SELECT group_concat((SELECT 2)) FROM x;`, Expected: []sql.Row{{"2,2,2,2,2"}}, }, { Query: `SELECT group_concat(DISTINCT (SELECT 2)) FROM x;`, Expected: []sql.Row{{"2"}}, }, { Query: "SELECT group_concat(DISTINCT `attribute` ORDER BY `attribute` ASC) FROM t", Expected: []sql.Row{{"color,fabric,shape"}}, }, { Query: "SELECT group_concat(DISTINCT `attribute` ORDER BY `attribute` DESC) FROM t", Expected: []sql.Row{{"shape,fabric,color"}}, }, { Query: `SELECT group_concat(pk) FROM nulls`, Expected: []sql.Row{{nil}}, }, { Query: `SELECT group_concat((SELECT * FROM t LIMIT 1)) from t`, ExpectedErr: sql.ErrInvalidOperandColumns, }, { Query: `SELECT group_concat((SELECT * FROM x)) from t`, ExpectedErr: sql.ErrExpectedSingleRow, }, { Query: "SELECT group_concat(`attribute`) FROM t where o_id=2 order by attribute", Expected: []sql.Row{{"color,fabric"}}, }, { Query: "SELECT group_concat(DISTINCT `attribute` ORDER BY value DESC SEPARATOR ';') FROM t group by o_id order by o_id asc", Expected: []sql.Row{{"fabric;color"}, {"shape;color"}}, }, { Query: "SELECT group_concat(o_id) FROM t WHERE `attribute`='color' order by o_id", Expected: []sql.Row{{"2,3"}}, }, }, }, { Name: "ALTER TABLE ... ALTER COLUMN SET / DROP DEFAULT", SetUpScript: []string{ "CREATE TABLE test (pk BIGINT PRIMARY KEY, v1 BIGINT NOT NULL DEFAULT 88);", }, Assertions: []ScriptTestAssertion{ { Query: "INSERT INTO test (pk) VALUES (1);", Expected: []sql.Row{{sql.NewOkResult(1)}}, }, { Query: "SELECT * FROM test;", Expected: []sql.Row{{1, 88}}, }, { Query: "ALTER TABLE test ALTER v1 SET DEFAULT (CONVERT('42', SIGNED));", Expected: []sql.Row{}, }, { Query: "INSERT INTO test (pk) VALUES (2);", Expected: []sql.Row{{sql.NewOkResult(1)}}, }, { Query: "SELECT * FROM test;", Expected: []sql.Row{{1, 88}, {2, 42}}, }, { Query: "ALTER TABLE test ALTER v2 SET DEFAULT 1;", ExpectedErr: sql.ErrTableColumnNotFound, }, { Query: "ALTER TABLE test ALTER v1 DROP DEFAULT;", Expected: []sql.Row{}, }, { Query: "INSERT INTO test (pk) VALUES (3);", ExpectedErr: sql.ErrInsertIntoNonNullableDefaultNullColumn, }, { Query: "ALTER TABLE test ALTER v2 DROP DEFAULT;", ExpectedErr: sql.ErrTableColumnNotFound, }, { Query: "SELECT * FROM test;", Expected: []sql.Row{{1, 88}, {2, 42}}, }, { Query: "ALTER TABLE test ALTER v1 SET DEFAULT 100, alter v1 DROP DEFAULT", Expected: []sql.Row{}, }, { Query: "INSERT INTO test (pk) VALUES (2);", ExpectedErr: sql.ErrInsertIntoNonNullableDefaultNullColumn, }, { Query: "ALTER TABLE test ALTER v1 SET DEFAULT 100, alter v1 SET DEFAULT 200", Expected: []sql.Row{}, }, { Query: "ALTER TABLE test DROP COLUMN v1, alter v1 SET DEFAULT 5000", ExpectedErr: sql.ErrTableColumnNotFound, }, { Query: "DESCRIBE test", Expected: []sql.Row{ {"pk", "bigint", "NO", "PRI", "", ""}, {"v1", "bigint", "NO", "", "200", ""}, }, }, }, }, { Name: "ALTER TABLE ... ALTER ADD CHECK / DROP CHECK", SetUpScript: []string{ "CREATE TABLE test (pk BIGINT PRIMARY KEY, v1 BIGINT NOT NULL DEFAULT 88);", }, Assertions: []ScriptTestAssertion{ { Query: "ALTER TABLE test ADD CONSTRAINT cx CHECK (v1 < 100)", Expected: []sql.Row{}, }, { Query: "ALTER TABLE test DROP CHECK cx, ADD CHECK (v1 < 50)", Expected: []sql.Row{}, }, { Query: "INSERT INTO test VALUES (1, 99)", ExpectedErr: sql.ErrCheckConstraintViolated, }, { Query: "INSERT INTO test VALUES (2, 2)", Expected: []sql.Row{{sql.NewOkResult(1)}}, }, }, }, { Name: "ALTER TABLE AUTO INCREMENT no-ops on table with no original auto increment key", SetUpScript: []string{ "CREATE table test (pk int primary key)", "ALTER TABLE `test` auto_increment = 2;", "INSERT INTO test VALUES (1)", }, Assertions: []ScriptTestAssertion{ { Query: "SELECT * FROM test", Expected: []sql.Row{{1}}, }, }, }, { Name: "ALTER TABLE MODIFY column with UNIQUE KEY", SetUpScript: []string{ "CREATE table test (pk int primary key, uk int unique)", "ALTER TABLE `test` MODIFY column uk int auto_increment", }, Assertions: []ScriptTestAssertion{ { Query: "describe test", Expected: []sql.Row{ {"pk", "int", "NO", "PRI", "", ""}, {"uk", "int", "YES", "UNI", "", "auto_increment"}, }, }, }, }, { Name: "ALTER TABLE MODIFY column with KEY", SetUpScript: []string{ "CREATE table test (pk int primary key, mk int, index (mk))", "ALTER TABLE `test` MODIFY column mk int auto_increment", }, Assertions: []ScriptTestAssertion{ { Query: "describe test", Expected: []sql.Row{ {"pk", "int", "NO", "PRI", "", ""}, {"mk", "int", "YES", "MUL", "", "auto_increment"}, }, }, }, }, { Name: "ALTER TABLE AUTO INCREMENT no-ops on table with no original auto increment key", SetUpScript: []string{ "CREATE table test (pk int primary key)", "ALTER TABLE `test` auto_increment = 2;", "INSERT INTO test VALUES (1)", }, Assertions: []ScriptTestAssertion{ { Query: "SELECT * FROM test", Expected: []sql.Row{{1}}, }, }, }, { Name: "Run through some complex queries with DISTINCT and aggregates", SetUpScript: []string{ "CREATE TABLE tab1(col0 INTEGER, col1 INTEGER, col2 INTEGER)", "CREATE TABLE tab2(col0 INTEGER, col1 INTEGER, col2 INTEGER)", "INSERT INTO tab1 VALUES(51,14,96)", "INSERT INTO tab1 VALUES(85,5,59)", "INSERT INTO tab1 VALUES(91,47,68)", "INSERT INTO tab2 VALUES(64,77,40)", "INSERT INTO tab2 VALUES(75,67,58)", "INSERT INTO tab2 VALUES(46,51,23)", "CREATE TABLE mytable (pk int, v1 int)", "INSERT INTO mytable VALUES(1,1)", "INSERT INTO mytable VALUES(1,1)", "INSERT INTO mytable VALUES(2,2)", "INSERT INTO mytable VALUES(1,2)", }, Assertions: []ScriptTestAssertion{ { Query: "SELECT - SUM( DISTINCT - - 71 ) AS col2 FROM tab2 cor0", Expected: []sql.Row{{float64(-71)}}, }, { Query: "SELECT - SUM ( DISTINCT - - 71 ) AS col2 FROM tab2 cor0", Expected: []sql.Row{{float64(-71)}}, }, { Query: "SELECT + MAX( DISTINCT ( - col0 ) ) FROM tab1 AS cor0", Expected: []sql.Row{{-51}}, }, { Query: "SELECT SUM( DISTINCT + col1 ) * - 22 - - ( - COUNT( * ) ) col0 FROM tab1 AS cor0", Expected: []sql.Row{{float64(-1455)}}, }, { Query: "SELECT MIN (DISTINCT col1) from tab1 GROUP BY col0 ORDER BY col0", Expected: []sql.Row{{14}, {5}, {47}}, }, { Query: "SELECT SUM (DISTINCT col1) from tab1 GROUP BY col0 ORDER BY col0", Expected: []sql.Row{{float64(14)}, {float64(5)}, {float64(47)}}, }, { Query: "SELECT pk, SUM(DISTINCT v1), MAX(v1) FROM mytable GROUP BY pk ORDER BY pk", Expected: []sql.Row{{int64(1), float64(3), int64(2)}, {int64(2), float64(2), int64(2)}}, }, { Query: "SELECT pk, MIN(DISTINCT v1), MAX(DISTINCT v1) FROM mytable GROUP BY pk ORDER BY pk", Expected: []sql.Row{{int64(1), int64(1), int64(2)}, {int64(2), int64(2), int64(2)}}, }, { Query: "SELECT SUM(DISTINCT pk * v1) from mytable", Expected: []sql.Row{{float64(7)}}, }, { Query: "SELECT SUM(DISTINCT POWER(v1, 2)) FROM mytable", Expected: []sql.Row{{float64(5)}}, }, { Query: "SELECT + + 97 FROM tab1 GROUP BY tab1.col1", Expected: []sql.Row{{97}, {97}, {97}}, }, { Query: "SELECT rand(10) FROM tab1 GROUP BY tab1.col1", Expected: []sql.Row{{0.5660920659323543}, {0.5660920659323543}, {0.5660920659323543}}, }, { Query: "SELECT ALL - cor0.col0 * + cor0.col0 AS col2 FROM tab1 AS cor0 GROUP BY cor0.col0", Expected: []sql.Row{{-2601}, {-7225}, {-8281}}, }, { Query: "SELECT cor0.col0 * cor0.col0 + cor0.col0 AS col2 FROM tab1 AS cor0 GROUP BY cor0.col0 order by 1", Expected: []sql.Row{{2652}, {7310}, {8372}}, }, { Query: "SELECT - floor(cor0.col0) * ceil(cor0.col0) AS col2 FROM tab1 AS cor0 GROUP BY cor0.col0", Expected: []sql.Row{{-2601}, {-7225}, {-8281}}, }, { Query: "SELECT col0 FROM tab1 AS cor0 GROUP BY cor0.col0", Expected: []sql.Row{{51}, {85}, {91}}, }, { Query: "SELECT - cor0.col0 FROM tab1 AS cor0 GROUP BY cor0.col0", Expected: []sql.Row{{-51}, {-85}, {-91}}, }, { Query: "SELECT col0 BETWEEN 2 and 4 from tab1 group by col0", Expected: []sql.Row{{false}, {false}, {false}}, }, { Query: "SELECT col0, col1 FROM tab1 GROUP by col0;", ExpectedErr: analyzer.ErrValidationGroupBy, }, { Query: "SELECT col0, floor(col1) FROM tab1 GROUP by col0;", ExpectedErr: analyzer.ErrValidationGroupBy, }, { Query: "SELECT floor(cor0.col1) * ceil(cor0.col0) AS col2 FROM tab1 AS cor0 GROUP BY cor0.col0", ExpectedErr: analyzer.ErrValidationGroupBy, }, }, }, { Name: "Nested Subquery projections (NTC)", SetUpScript: []string{ `CREATE TABLE dcim_site (id char(32) NOT NULL,created date,last_updated datetime,_custom_field_data json NOT NULL,name varchar(100) NOT NULL,_name varchar(100) NOT NULL,slug varchar(100) NOT NULL,facility varchar(50) NOT NULL,asn bigint,time_zone varchar(63) NOT NULL,description varchar(200) NOT NULL,physical_address varchar(200) NOT NULL,shipping_address varchar(200) NOT NULL,latitude decimal(8,6),longitude decimal(9,6),contact_name varchar(50) NOT NULL,contact_phone varchar(20) NOT NULL,contact_email varchar(254) NOT NULL,comments longtext NOT NULL,region_id char(32),status_id char(32),tenant_id char(32),PRIMARY KEY (id),KEY dcim_site_region_id_45210932 (region_id),KEY dcim_site_status_id_e6a50f56 (status_id),KEY dcim_site_tenant_id_15e7df63 (tenant_id),UNIQUE KEY name (name),UNIQUE KEY slug (slug)) ENGINE=InnoDB DEFAULT CHARSET=utf8mb4 COLLATE=utf8mb4_0900_bin;`, `CREATE TABLE dcim_rackgroup (id char(32) NOT NULL,created date,last_updated datetime,_custom_field_data json NOT NULL,name varchar(100) NOT NULL,slug varchar(100) NOT NULL,description varchar(200) NOT NULL,lft int unsigned NOT NULL,rght int unsigned NOT NULL,tree_id int unsigned NOT NULL,level int unsigned NOT NULL,parent_id char(32),site_id char(32) NOT NULL,PRIMARY KEY (id),KEY dcim_rackgroup_parent_id_cc315105 (parent_id),KEY dcim_rackgroup_site_id_13520e89 (site_id),KEY dcim_rackgroup_slug_3f4582a7 (slug),KEY dcim_rackgroup_tree_id_9c2ad6f4 (tree_id),UNIQUE KEY site_idname (site_id,name),UNIQUE KEY site_idslug (site_id,slug),CONSTRAINT dcim_rackgroup_parent_id_cc315105_fk_dcim_rackgroup_id FOREIGN KEY (parent_id) REFERENCES dcim_rackgroup (id),CONSTRAINT dcim_rackgroup_site_id_13520e89_fk_dcim_site_id FOREIGN KEY (site_id) REFERENCES dcim_site (id)) ENGINE=InnoDB DEFAULT CHARSET=utf8mb4 COLLATE=utf8mb4_0900_bin;`, `CREATE TABLE dcim_rack (id char(32) NOT NULL,created date,last_updated datetime,_custom_field_data json NOT NULL,name varchar(100) NOT NULL,_name varchar(100) NOT NULL,facility_id varchar(50),serial varchar(50) NOT NULL,asset_tag varchar(50),type varchar(50) NOT NULL,width smallint unsigned NOT NULL,u_height smallint unsigned NOT NULL,desc_units tinyint NOT NULL,outer_width smallint unsigned,outer_depth smallint unsigned,outer_unit varchar(50) NOT NULL,comments longtext NOT NULL,group_id char(32),role_id char(32),site_id char(32) NOT NULL,status_id char(32),tenant_id char(32),PRIMARY KEY (id),UNIQUE KEY asset_tag (asset_tag),KEY dcim_rack_group_id_44e90ea9 (group_id),KEY dcim_rack_role_id_62d6919e (role_id),KEY dcim_rack_site_id_403c7b3a (site_id),KEY dcim_rack_status_id_ee3dee3e (status_id),KEY dcim_rack_tenant_id_7cdf3725 (tenant_id),UNIQUE KEY group_idfacility_id (group_id,facility_id),UNIQUE KEY group_idname (group_id,name),CONSTRAINT dcim_rack_group_id_44e90ea9_fk_dcim_rackgroup_id FOREIGN KEY (group_id) REFERENCES dcim_rackgroup (id),CONSTRAINT dcim_rack_site_id_403c7b3a_fk_dcim_site_id FOREIGN KEY (site_id) REFERENCES dcim_site (id)) ENGINE=InnoDB DEFAULT CHARSET=utf8mb4 COLLATE=utf8mb4_0900_bin;`, `INSERT INTO dcim_site (id, created, last_updated, _custom_field_data, status_id, name, _name, slug, region_id, tenant_id, facility, asn, time_zone, description, physical_address, shipping_address, latitude, longitude, contact_name, contact_phone, contact_email, comments) VALUES ('f0471f313b694d388c8ec39d9590e396', '2021-05-20', '2021-05-20 18:51:46.416695', '{}', NULL, 'Site 1', 'Site 00000001', 'site-1', NULL, NULL, '', NULL, '', '', '', '', NULL, NULL, '', '', '', '')`, `INSERT INTO dcim_site (id, created, last_updated, _custom_field_data, status_id, name, _name, slug, region_id, tenant_id, facility, asn, time_zone, description, physical_address, shipping_address, latitude, longitude, contact_name, contact_phone, contact_email, comments) VALUES ('442bab8b517149ab87207e8fb5ba1569', '2021-05-20', '2021-05-20 18:51:47.333720', '{}', NULL, 'Site 2', 'Site 00000002', 'site-2', NULL, NULL, '', NULL, '', '', '', '', NULL, NULL, '', '', '', '')`, `INSERT INTO dcim_rackgroup (id, created, last_updated, _custom_field_data, name, slug, site_id, parent_id, description, lft, rght, tree_id, level) VALUES ('5c107f979f434bf7a7820622f18a5211', '2021-05-20', '2021-05-20 18:51:48.150116', '{}', 'Parent Rack Group 1', 'parent-rack-group-1', 'f0471f313b694d388c8ec39d9590e396', NULL, '', 1, 2, 1, 0)`, `INSERT INTO dcim_rackgroup (id, created, last_updated, _custom_field_data, name, slug, site_id, parent_id, description, lft, rght, tree_id, level) VALUES ('6707c20336a2406da6a9d394477f7e8c', '2021-05-20', '2021-05-20 18:51:48.969713', '{}', 'Parent Rack Group 2', 'parent-rack-group-2', '442bab8b517149ab87207e8fb5ba1569', NULL, '', 1, 2, 2, 0)`, `INSERT INTO dcim_rackgroup (id, created, last_updated, _custom_field_data, name, slug, site_id, parent_id, description, lft, rght, tree_id, level) VALUES ('6bc0d9b1affe46918b09911359241db6', '2021-05-20', '2021-05-20 18:51:50.566160', '{}', 'Rack Group 1', 'rack-group-1', 'f0471f313b694d388c8ec39d9590e396', '5c107f979f434bf7a7820622f18a5211', '', 2, 3, 1, 1)`, `INSERT INTO dcim_rackgroup (id, created, last_updated, _custom_field_data, name, slug, site_id, parent_id, description, lft, rght, tree_id, level) VALUES ('a773cac9dc9842228cdfd8c97a67136e', '2021-05-20', '2021-05-20 18:51:52.126952', '{}', 'Rack Group 2', 'rack-group-2', 'f0471f313b694d388c8ec39d9590e396', '5c107f979f434bf7a7820622f18a5211', '', 4, 5, 1, 1)`, `INSERT INTO dcim_rackgroup (id, created, last_updated, _custom_field_data, name, slug, site_id, parent_id, description, lft, rght, tree_id, level) VALUES ('a35a843eb181404bb9da2126c6580977', '2021-05-20', '2021-05-20 18:51:53.706000', '{}', 'Rack Group 3', 'rack-group-3', 'f0471f313b694d388c8ec39d9590e396', '5c107f979f434bf7a7820622f18a5211', '', 6, 7, 1, 1)`, `INSERT INTO dcim_rackgroup (id, created, last_updated, _custom_field_data, name, slug, site_id, parent_id, description, lft, rght, tree_id, level) VALUES ('f09a02c95b064533b823e25374f5962a', '2021-05-20', '2021-05-20 18:52:03.037056', '{}', 'Test Rack Group 4', 'test-rack-group-4', '442bab8b517149ab87207e8fb5ba1569', '6707c20336a2406da6a9d394477f7e8c', '', 2, 3, 2, 1)`, `INSERT INTO dcim_rackgroup (id, created, last_updated, _custom_field_data, name, slug, site_id, parent_id, description, lft, rght, tree_id, level) VALUES ('ecff5b528c5140d4a58f1b24a1c80ebc', '2021-05-20', '2021-05-20 18:52:05.390373', '{}', 'Test Rack Group 5', 'test-rack-group-5', '442bab8b517149ab87207e8fb5ba1569', '6707c20336a2406da6a9d394477f7e8c', '', 4, 5, 2, 1)`, `INSERT INTO dcim_rackgroup (id, created, last_updated, _custom_field_data, name, slug, site_id, parent_id, description, lft, rght, tree_id, level) VALUES ('d31b3772910e4418bdd5725d905e2699', '2021-05-20', '2021-05-20 18:52:07.758547', '{}', 'Test Rack Group 6', 'test-rack-group-6', '442bab8b517149ab87207e8fb5ba1569', '6707c20336a2406da6a9d394477f7e8c', '', 6, 7, 2, 1)`, `INSERT INTO dcim_rack (id,created,last_updated,_custom_field_data,name,_name,facility_id,serial,asset_tag,type,width,u_height,desc_units,outer_width,outer_depth,outer_unit,comments,group_id,role_id,site_id,status_id,tenant_id) VALUES ('abc123', '2021-05-20', '2021-05-20 18:51:48.150116', '{}', "name", "name", "facility", "serial", "assettag", "type", 1, 1, 1, 1, 1, "outer units", "comment", "6bc0d9b1affe46918b09911359241db6", "role", "f0471f313b694d388c8ec39d9590e396", "status", "tenant")`, }, Assertions: []ScriptTestAssertion{ { Query: `SELECT (( SELECT COUNT(*) FROM dcim_rack WHERE group_id IN ( SELECT m2.id FROM dcim_rackgroup m2 WHERE m2.tree_id = dcim_rackgroup.tree_id AND m2.lft BETWEEN dcim_rackgroup.lft AND dcim_rackgroup.rght ) )) AS rack_count, dcim_rackgroup.id, dcim_rackgroup._custom_field_data, dcim_rackgroup.name, dcim_rackgroup.slug, dcim_rackgroup.site_id, dcim_rackgroup.parent_id, dcim_rackgroup.description, dcim_rackgroup.lft, dcim_rackgroup.rght, dcim_rackgroup.tree_id, dcim_rackgroup.level FROM dcim_rackgroup order by 2 limit 1`, Expected: []sql.Row{{1, "5c107f979f434bf7a7820622f18a5211", sql.JSONDocument{Val: map[string]interface{}{}}, "Parent Rack Group 1", "parent-rack-group-1", "f0471f313b694d388c8ec39d9590e396", interface{}(nil), "", uint64(1), uint64(2), uint64(1), uint64(0)}}, }, }, }, { Name: "CREATE TABLE SELECT Queries", SetUpScript: []string{ `CREATE TABLE t1 (pk int PRIMARY KEY, v1 varchar(10))`, `INSERT INTO t1 VALUES (1,"1"), (2,"2"), (3,"3")`, `CREATE TABLE t2 AS SELECT * FROM t1`, `CREATE TABLE t3 AS SELECT pk FROM t1`, `CREATE TABLE t4 AS SELECT pk, v1 FROM t1`, `CREATE TABLE t5 SELECT * FROM t1 ORDER BY pk LIMIT 1`, }, Assertions: []ScriptTestAssertion{ { Query: `SELECT * FROM t2`, Expected: []sql.Row{{1, "1"}, {2, "2"}, {3, "3"}}, }, { Query: `SELECT * FROM t3`, Expected: []sql.Row{{1}, {2}, {3}}, }, { Query: `SELECT * FROM t4`, Expected: []sql.Row{{1, "1"}, {2, "2"}, {3, "3"}}, }, { Query: `SELECT * FROM t5`, Expected: []sql.Row{{1, "1"}}, }, { Query: `CREATE TABLE test SELECT * FROM t1`, Expected: []sql.Row{sql.Row{sql.OkResult{ RowsAffected: 3, InsertID: 0, Info: nil, }}}, }, }, }, { Name: "Issue #499", SetUpScript: []string{ "CREATE TABLE test (time TIMESTAMP, value DOUBLE);", `INSERT INTO test VALUES ("2021-07-04 10:00:00", 1.0), ("2021-07-03 10:00:00", 2.0), ("2021-07-02 10:00:00", 3.0), ("2021-07-01 10:00:00", 4.0);`, }, Assertions: []ScriptTestAssertion{ { Query: `SELECT UNIX_TIMESTAMP(time) DIV 60 * 60 AS "time", avg(value) AS "value" FROM test GROUP BY 1 ORDER BY UNIX_TIMESTAMP(time) DIV 60 * 60`, Expected: []sql.Row{ {1625133600, 4.0}, {1625220000, 3.0}, {1625306400, 2.0}, {1625392800, 1.0}, }, }, }, SkipPrepared: true, }, { Name: "WHERE clause considers ENUM/SET types for comparisons", SetUpScript: []string{ "CREATE TABLE test (pk BIGINT PRIMARY KEY, v1 ENUM('a', 'b', 'c'), v2 SET('a', 'b', 'c'));", "INSERT INTO test VALUES (1, 2, 2), (2, 1, 1);", }, Assertions: []ScriptTestAssertion{ { Query: "SELECT * FROM test;", Expected: []sql.Row{{1, uint16(2), uint64(2)}, {2, uint16(1), uint64(1)}}, }, { Query: "UPDATE test SET v1 = 3 WHERE v1 = 2;", Expected: []sql.Row{{sql.OkResult{RowsAffected: 1, InsertID: 0, Info: plan.UpdateInfo{Matched: 1, Updated: 1}}}}, }, { Query: "SELECT * FROM test;", Expected: []sql.Row{{1, uint16(3), uint64(2)}, {2, uint16(1), uint64(1)}}, }, { Query: "UPDATE test SET v2 = 3 WHERE 2 = v2;", Expected: []sql.Row{{sql.OkResult{RowsAffected: 1, InsertID: 0, Info: plan.UpdateInfo{Matched: 1, Updated: 1}}}}, }, { Query: "SELECT * FROM test;", Expected: []sql.Row{{1, uint16(3), uint64(3)}, {2, uint16(1), uint64(1)}}, }, }, }, { Name: "Slightly more complex example for the Exists Clause", SetUpScript: []string{ "create table store(store_id int, item_id int, primary key (store_id, item_id))", "create table items(item_id int primary key, price int)", "insert into store values (0, 1), (0,2),(0,3),(1,2),(1,4),(2,1)", "insert into items values (1, 10), (2, 20), (3, 30),(4,40)", }, Assertions: []ScriptTestAssertion{ { Query: "SELECT * from store WHERE EXISTS (SELECT price from items where price > 10 and store.item_id = items.item_id)", Expected: []sql.Row{{0, 2}, {0, 3}, {1, 2}, {1, 4}}, }, }, }, { Name: "Simple Update Join test that manipulates two tables", SetUpScript: []string{ "CREATE TABLE test (pk int primary key);", `CREATE TABLE test2 (pk int primary key, val int);`, `INSERT into test values (0),(1),(2),(3)`, `INSERT into test2 values (0, 0),(1, 1),(2, 2),(3, 3)`, `CREATE TABLE test3(k int, val int, primary key (k, val))`, `INSERT into test3 values (1,2),(1,3),(1,4)`, }, Assertions: []ScriptTestAssertion{ { Query: `update test2 inner join (select * from test3 order by val) as t3 on test2.pk = t3.k SET test2.val=t3.val`, Expected: []sql.Row{{sql.OkResult{RowsAffected: 1, Info: plan.UpdateInfo{ Matched: 1, Updated: 1, Warnings: 0, }}}}, }, { Query: "SELECT val FROM test2 where pk = 1", Expected: []sql.Row{ {2}, }, }, { Query: `update test inner join test2 on test.pk = test2.pk SET test.pk=test.pk*10, test2.pk = test2.pk * 4 where test.pk < 10;`, Expected: []sql.Row{{sql.OkResult{RowsAffected: 6, Info: plan.UpdateInfo{ Matched: 6, Updated: 6, Warnings: 0, }}}}, }, { Query: "SELECT * FROM test", Expected: []sql.Row{ {0}, {10}, {20}, {30}, }, }, { Query: "SELECT * FROM test2", Expected: []sql.Row{ {0, 0}, {4, 2}, {8, 2}, {12, 3}, }, }, }, }, { Name: "Partial indexes are used and return the expected result", SetUpScript: []string{ "CREATE TABLE test (pk BIGINT PRIMARY KEY, v1 BIGINT, v2 BIGINT, v3 BIGINT, INDEX vx (v3, v2, v1));", "INSERT INTO test VALUES (1,2,3,4), (2,3,4,5), (3,4,5,6), (4,5,6,7), (5,6,7,8);", }, Assertions: []ScriptTestAssertion{ { Query: "EXPLAIN SELECT * FROM test WHERE v3 = 4;", Expected: []sql.Row{{"Projected table access on [pk v1 v2 v3]"}, {" └─ IndexedTableAccess(test on [test.v3,test.v2,test.v1] with ranges: [{[4, 4], (-∞, ∞), (-∞, ∞)}])"}}, }, { Query: "SELECT * FROM test WHERE v3 = 4;", Expected: []sql.Row{{1, 2, 3, 4}}, }, { Query: "EXPLAIN SELECT * FROM test WHERE v3 = 8 AND v2 = 7;", Expected: []sql.Row{ {"Projected table access on [pk v1 v2 v3]"}, {" └─ IndexedTableAccess(test on [test.v3,test.v2,test.v1] with ranges: [{[8, 8], [7, 7], (-∞, ∞)}])"}}, }, { Query: "SELECT * FROM test WHERE v3 = 8 AND v2 = 7;", Expected: []sql.Row{{5, 6, 7, 8}}, }, { Query: "EXPLAIN SELECT * FROM test WHERE v3 >= 6 AND v2 >= 6;", Expected: []sql.Row{ {"Projected table access on [pk v1 v2 v3]"}, {" └─ IndexedTableAccess(test on [test.v3,test.v2,test.v1] with ranges: [{[6, ∞), [6, ∞), (-∞, ∞)}])"}}, }, { Query: "SELECT * FROM test WHERE v3 >= 6 AND v2 >= 6;", Expected: []sql.Row{{4, 5, 6, 7}, {5, 6, 7, 8}}, }, { Query: "EXPLAIN SELECT * FROM test WHERE v3 = 7 AND v2 >= 6;", Expected: []sql.Row{ {"Projected table access on [pk v1 v2 v3]"}, {" └─ IndexedTableAccess(test on [test.v3,test.v2,test.v1] with ranges: [{[7, 7], [6, ∞), (-∞, ∞)}])"}}, }, { Query: "SELECT * FROM test WHERE v3 = 7 AND v2 >= 6;", Expected: []sql.Row{{4, 5, 6, 7}}, }, }, }, { Name: "Multiple indexes on the same columns in a different order", SetUpScript: []string{ "CREATE TABLE test (pk BIGINT PRIMARY KEY, v1 BIGINT, v2 BIGINT, v3 BIGINT, INDEX v123 (v1, v2, v3), INDEX v321 (v3, v2, v1), INDEX v132 (v1, v3, v2));", "INSERT INTO test VALUES (1,2,3,4), (2,3,4,5), (3,4,5,6), (4,5,6,7), (5,6,7,8);", }, Assertions: []ScriptTestAssertion{ { Query: "EXPLAIN SELECT * FROM test WHERE v1 = 2 AND v2 > 1;", Expected: []sql.Row{ {"Projected table access on [pk v1 v2 v3]"}, {" └─ IndexedTableAccess(test on [test.v1,test.v2,test.v3] with ranges: [{[2, 2], (1, ∞), (-∞, ∞)}])"}}, }, { Query: "SELECT * FROM test WHERE v1 = 2 AND v2 > 1;", Expected: []sql.Row{{1, 2, 3, 4}}, }, { Query: "EXPLAIN SELECT * FROM test WHERE v2 = 4 AND v3 > 1;", Expected: []sql.Row{ {"Projected table access on [pk v1 v2 v3]"}, {" └─ IndexedTableAccess(test on [test.v3,test.v2,test.v1] with ranges: [{(1, ∞), [4, 4], (-∞, ∞)}])"}}, }, { Query: "SELECT * FROM test WHERE v2 = 4 AND v3 > 1;", Expected: []sql.Row{{2, 3, 4, 5}}, }, { Query: "EXPLAIN SELECT * FROM test WHERE v3 = 6 AND v1 > 1;", Expected: []sql.Row{ {"Projected table access on [pk v1 v2 v3]"}, {" └─ IndexedTableAccess(test on [test.v1,test.v3,test.v2] with ranges: [{(1, ∞), [6, 6], (-∞, ∞)}])"}}, }, { Query: "SELECT * FROM test WHERE v3 = 6 AND v1 > 1;", Expected: []sql.Row{{3, 4, 5, 6}}, }, { Query: "EXPLAIN SELECT * FROM test WHERE v1 = 5 AND v3 <= 10 AND v2 >= 1;", Expected: []sql.Row{ {"Projected table access on [pk v1 v2 v3]"}, {" └─ IndexedTableAccess(test on [test.v1,test.v2,test.v3] with ranges: [{[5, 5], [1, ∞), (-∞, 10]}])"}}, }, { Query: "SELECT * FROM test WHERE v1 = 5 AND v3 <= 10 AND v2 >= 1;", Expected: []sql.Row{{4, 5, 6, 7}}, }, }, }, { Name: "Ensure proper DECIMAL support (found by fuzzer)", SetUpScript: []string{ "CREATE TABLE `GzaKtwgIya` (`K7t5WY` DECIMAL(64,5), `qBjVrN` VARBINARY(1000), `PvqQtc` SET('c3q6y','kxMqhfkK','XlRI8','dF0N63H','hMPjt0KXRLwCGRr','27fi2s','1FSJ','NcPzIN','Za18lbIgxmZ','on4BKKXykVTbJ','WBfO','RMNG','Sd7','FDzbEO','cLRdLOj1y','syo4','Ul','jfsfDCx6s','yEW3','JyQcWFDl'), `1kv7el` FLOAT, `Y3vfRG` BLOB, `Ijq8CK` TINYTEXT, `tzeStN` MEDIUMINT, `Ak83FQ` BINARY(64), `8Nbp3L` DOUBLE, PRIMARY KEY (`K7t5WY`));", "REPLACE INTO `GzaKtwgIya` VALUES ('58567047399981325523662211357420045483361289734772861386428.89028','bvo5~Tt8%kMW2nm2!8HghaeulI6!pMadE+j-J2LeU1O1*-#@Lm8Ibh00bTYiA*H1Q8P1_kQq 24Rrd4@HeF%#7#C#U7%mqOMrQ0%!HVrGV1li.XyYa:7#3V^DtAMDTQ9 cY=07T4|DStrwy4.MAQxOG#1d#fcq+7675$y0e96-2@8-WlQ^p|%E!a^TV!Yj2_eqZZys1z:883l5I%zAT:i56K^T!cx#us $60Tb#gH$1#$P.709E#VrH9FbQ5QZK2hZUH!qUa4Xl8*I*0fT~oAha$8jU5AoWs+Uv!~:14Yq%pLXpP9RlZ:Gd1g|*$Qa.9*^K~YlYWVaxwY~_g6zOMpU$YijT+!_*m3=||cMNn#uN0!!OyCg~GTQlJ11+#@Ohqc7b#2|Jp2Aei56GOmq^I=7cQ=sQh~V.D^HzwK5~4E$QzFXfWNVN5J_w2b4dkR~bB~7F%=@R@9qE~e:-_RnoJcOLfBS@0:*hTIP$5ui|5Ea-l+qU4nx98X6rV2bLBxn8am@p~:xLF#T^_9kJVN76q^18=i *FJo.v-xA2GP==^C^Jz3yBF0OY4bIxC59Y#6G=$w:xh71kMxBcYJKf3+$Ci_uWx0P*AfFNne0_1E0Lwv#3J8vm:. 8Wo~F3VT:@w.t@w .JZz$bok9Tls7RGo=~4 Y$~iELr$s@53YuTPM8oqu!x*1%GswpJR=0K#qs00nW-1MqEUc:0wZv#X4qY^pzVDb:!:!yDhjhh+KIT%2%w@+t8c!f~o!%EnwBIr_OyzL6e1$-R8n0nWPU.toODd*|fW3H$9ZLc9!dMS:QfjI0M$nK 8aGvUVP@9kS~W#Y=Q%=37$@pAUkDTXkJo~-DRvCG6phPp*Xji@9|AEODHi+-6p%X4YM5Y3WasPHcZQ8QgTwi9 N=2RQD_MtVU~0J~3SAx*HrMlKvCPTswZq#q_96ny_A@7g!E2jyaxWFJD:C233onBdchW$WdAc.LZdZHYDR^uwZb9B9p-q.BkD1I',608583,'-7.276514330627342e-28','FN3O_E:$ 5S40T7^Vu1g!Ktn^N|4RE!9GnZiW5dG:%SJb5|SNuuI.d2^qnMY.Xn*_fRfk Eo7OhqY8OZ~pA0^ !2P.uN~r@pZ2!A0+4b*%nxO.tm%S6=$CZ9+c1zu-p $b:7:fOkC%@E3951st@2Q93~8hj:ZGeJ6S@nw-TAG+^lad37aB#xN*rD^9TO0|hleA#.Nh28S2PB72L*TxD0$|XE3S5eVVmbI*pkzE~lPecopX1fUyFj#LC+%~pjmab7^ Kdd4B%8I!ohOCQV.oiw++N|#W2=D4:_sK0@~kTTeNA8_+FMKRwro.M0| LdKHf-McKm0Z-R9+H%!9r l6%7UEB50yNH-ld%eW8!f=LKgZLc*TuTP2DA_o0izvzZokNp3ShR+PA7Fk* 1RcSt5KXe+8tLc+WGP','3RvfN2N.Q1tIffE965#2r=u_-4!u:9w!F1p7+mSsO8ckio|ib 1t@~GtgUkJX',1858932,'DJMaQcI=vS-Jk2L#^2N8qZcRpMJ2Ga!30A+@I!+35d-9bwVEVi5-~i.a%!KdoF5h','1.0354401044541863e+255');", "INSERT INTO `GzaKtwgIya` VALUES ('91198031969464085142628031466155813748261645250257051732159.65596','96Lu=focmodq4otVAUN6TD-F$@k^4443higo=KH!1WBDH9|vpEGdO* 1uF6yWjT4:7G|altXnWSv+d:c8Km8vL!b%-nuB8mAxO9E|a5N5#v@z!ij5ifeIEoZGXrhBJl.m*Rx-@%g~t:y$3Pp3Q7Bd3y$=YG%6yibqXWO9$SS+g=*6QzdSCzuR~@v!:.ATye0A@y~DG=uq!PaZd6wN7.2S Aq868-RN3RM61V#N+Qywqo=%iYV*554@h6GPKZ| pmNwQw=PywuyBhr*MHAOXV+u9_-#imKI-wT4gEcA1~lGg1cfL2IvhkwOXRhrjAx-8+R3#4!Ai J6SYP|YUuuGalJ_N8k_8K^~h!JyiH$0JbGQ4AOxO3-eW=BaopOd8FF1.cfFMK!tXR ^I15g:npOuZZO$Vq3yQ4bl4s$E9:t2^.4f.:I4_@u9_UI1ApBthJZNiv~o#*uhs9K@ufZ1YPJQY-pMj$v-lQ2#%=Uu!iEAO3%vQ^5YITKcWRk~$kd1H#F675r@P5#M%*F_xP3Js7$YuEC4YuQjZ A74tMw:KwQ8dR:k_ Sa85G~42-K3%:jk5G9csC@iW3nY|@-:_dg~5@J!FWF5F+nyBgz4fDpdkdk9^:_.t$A3W-C@^Ax.~o|Rq96_i%HeG*7jBjOGhY-e1k@aD@WW.@GmpGAI|T-84gZFG3BU9@#9lpL|U2YCEA.BEA%sxDZ Kw:n+d$Y!SZw0Iml$Bdtyr:02Np=DZpiI%$N9*U=%Jq#$P5BI60WOTK+UynVx9Dd**5q8y9^v+I|PPa#_2XheV5YQU.ONdQQNJxsiRaEl!*=xv4bTWj1wBH#_-eM3T',490529,'-8.419238802182018e+25','|WD!NpWJOfN+_Au 1y!|XF8l38#%%R5%$TRUEaFt%4ywKQ8 O1LD-3qRDrnHAXboH~0uivbo87f+V%=q9~Mvz1EIxsU!whSmPqtb9r*11346R_@L+H#@@Z9H-Dc6j%.D0o##m@B9o7jO#~N81ACI|f#J3z4dho:jc54Xws$8r%cxuov^1$w_58Fv2*.8qbAW$TF153A:8wwj4YIhkd#^Q7 |g7I0iQG0p+yE64rk!Pu!SA-z=ELtLNOCJBk_4!lV$izn%sB6JwM+uq~ 49I7','v|eUA_h2@%t~bn26ci8Ngjm@Lk*G=l2MhxhceV2V|ka#c',8150267,'nX-=1Q$3riw_jlukGuHmjodT_Y_SM$xRbEt$%$%hlIUF1+GpRp~U6JvRX^: k@n#','7.956726808353253e+267');", }, Assertions: []ScriptTestAssertion{ { Query: "DELETE FROM `GzaKtwgIya` WHERE `K7t5WY` = '58567047399981325523662211357420045483361289734772861386428.89028';", Expected: []sql.Row{{sql.NewOkResult(1)}}, }, { Query: "SELECT COUNT(*) FROM GzaKtwgIya", Expected: []sql.Row{{1}}, }, }, }, { Name: "JOIN on non-index-prefix columns do not panic (Dolt Issue #2366)", SetUpScript: []string{ "CREATE TABLE `player_season_stat_totals` (`player_id` int NOT NULL, `team_id` int NOT NULL, `season_id` int NOT NULL, `minutes` int, `games_started` int, `games_played` int, `2pm` int, `2pa` int, `3pm` int, `3pa` int, `ftm` int, `fta` int, `ast` int, `stl` int, `blk` int, `tov` int, `pts` int, `orb` int, `drb` int, `trb` int, `pf` int, `season_type_id` int NOT NULL, `league_id` int NOT NULL DEFAULT 0, PRIMARY KEY (`player_id`,`team_id`,`season_id`,`season_type_id`,`league_id`));", "CREATE TABLE `team_seasons` (`team_id` int NOT NULL, `league_id` int NOT NULL, `season_id` int NOT NULL, `prefix` varchar(100), `nickname` varchar(100), `abbreviation` varchar(100), `city` varchar(100), `state` varchar(100), `country` varchar(100), PRIMARY KEY (`team_id`,`league_id`,`season_id`));", "INSERT INTO player_season_stat_totals VALUES (1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1);", "INSERT INTO team_seasons VALUES (1,1,1,'','','','','','');", }, Assertions: []ScriptTestAssertion{ { Query: "SELECT stats.* FROM player_season_stat_totals stats LEFT JOIN team_seasons ON team_seasons.team_id = stats.team_id AND team_seasons.season_id = stats.season_id;", Expected: []sql.Row{{1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1}}, }, }, }, { Name: "Issue #709", SetUpScript: []string{ "create table a(id int primary key, v int , key (v));", }, }, { Name: "Show create table with various keys and constraints", SetUpScript: []string{ "create table t1(a int primary key, b varchar(10) not null default 'abc')", "alter table t1 add constraint ck1 check (b like '%abc%')", "create index t1b on t1(b)", "create table t2(c int primary key, d varchar(10))", "alter table t2 add constraint t2du unique (d)", "alter table t2 add constraint fk1 foreign key (d) references t1 (b)", "create table t3 (a int, b varchar(100), c datetime, primary key (b,a))", "create table t4 (a int default floor(1), b int default coalesce(a, 10))", }, Assertions: []ScriptTestAssertion{ { Query: "show create table t1", Expected: []sql.Row{ {"t1", "CREATE TABLE `t1` (\n" + " `a` int NOT NULL,\n" + " `b` varchar(10) NOT NULL DEFAULT \"abc\",\n" + " PRIMARY KEY (`a`),\n" + " KEY `t1b` (`b`),\n" + " CONSTRAINT `ck1` CHECK (`b` LIKE \"%abc%\")\n" + ") ENGINE=InnoDB DEFAULT CHARSET=utf8mb4 COLLATE=utf8mb4_0900_bin"}, }, }, { Query: "show create table t2", Expected: []sql.Row{ {"t2", "CREATE TABLE `t2` (\n" + " `c` int NOT NULL,\n" + " `d` varchar(10),\n" + " PRIMARY KEY (`c`),\n" + " UNIQUE KEY `d` (`d`),\n" + " CONSTRAINT `fk1` FOREIGN KEY (`d`) REFERENCES `t1` (`b`)\n" + ") ENGINE=InnoDB DEFAULT CHARSET=utf8mb4 COLLATE=utf8mb4_0900_bin"}, }, }, { Query: "show create table t3", Expected: []sql.Row{ {"t3", "CREATE TABLE `t3` (\n" + " `a` int NOT NULL,\n" + " `b` varchar(100) NOT NULL,\n" + " `c` datetime,\n" + " PRIMARY KEY (`b`,`a`)\n" + ") ENGINE=InnoDB DEFAULT CHARSET=utf8mb4 COLLATE=utf8mb4_0900_bin"}, }, }, { Query: "show create table t4", Expected: []sql.Row{ {"t4", "CREATE TABLE `t4` (\n" + " `a` int DEFAULT (FLOOR(1)),\n" + " `b` int DEFAULT (coalesce(a, 10))\n" + ") ENGINE=InnoDB DEFAULT CHARSET=utf8mb4 COLLATE=utf8mb4_0900_bin"}, }, }, }, }, { Name: "table with defaults, insert with on duplicate key update", SetUpScript: []string{ "create table t (a int primary key, b int default 100);", "insert into t values (1, 1), (2, 2)", }, Assertions: []ScriptTestAssertion{ { Query: "insert into t values (1, 10) on duplicate key update b = 10", Expected: []sql.Row{{sql.NewOkResult(2)}}, }, }, }, { Name: "delete from table with misordered pks", SetUpScript: []string{ "create table a (x int, y int, z int, primary key (z,x))", "insert into a values (0,1,2), (3,4,5)", }, Assertions: []ScriptTestAssertion{ { Query: "SELECT count(*) FROM a where x = 0", Expected: []sql.Row{ {1}, }, }, { Query: "delete from a where x = 0", Expected: []sql.Row{{sql.NewOkResult(1)}}, }, { Query: "SELECT * FROM a where x = 0", Expected: []sql.Row{}, }, }, }, { Name: "recreate primary key rebuilds secondary indexes", SetUpScript: []string{ "create table a (x int, y int, z int, primary key (x,y,z), index idx1 (y))", "insert into a values (1,2,3), (4,5,6), (7,8,9)", "alter table a drop primary key", "alter table a add primary key (y,z,x)", }, Assertions: []ScriptTestAssertion{ { Query: "delete from a where y = 2", Expected: []sql.Row{{sql.NewOkResult(1)}}, }, { Query: "delete from a where y = 2", Expected: []sql.Row{{sql.NewOkResult(0)}}, }, { Query: "select * from a where y = 2", Expected: []sql.Row{}, }, { Query: "select * from a where y = 5", Expected: []sql.Row{{4, 5, 6}}, }, }, }, { Name: "Handle hex number to binary conversion", SetUpScript: []string{ "CREATE TABLE hex_nums1 (pk BIGINT PRIMARY KEY, v1 INT, v2 BIGINT UNSIGNED, v3 DOUBLE, v4 BINARY(32));", "INSERT INTO hex_nums1 values (1, 0x7ED0599B, 0x765a8ce4ce74b187, 0xF753AD20B0C4, 0x148aa875c3cdb9af8919493926a3d7c6862fec7f330152f400c0aecb4467508a);", "CREATE TABLE hex_nums2 (pk BIGINT PRIMARY KEY, v1 VARBINARY(255), v2 BLOB);", "INSERT INTO hex_nums2 values (1, 0x765a8ce4ce74b187, 0x148aa875c3cdb9af8919493926a3d7c6862fec7f330152f400c0aecb4467508a);", }, Assertions: []ScriptTestAssertion{ { Query: "SELECT v1, v2, v3, hex(v4) FROM hex_nums1;", Expected: []sql.Row{{2127583643, uint64(8528283758723641735), float64(271938758947012), "148AA875C3CDB9AF8919493926A3D7C6862FEC7F330152F400C0AECB4467508A"}}, }, { Query: "SELECT hex(v1), hex(v2), hex(v3), hex(v4) FROM hex_nums1;", Expected: []sql.Row{{"7ED0599B", "765A8CE4CE74B187", "F753AD20B0C4", "148AA875C3CDB9AF8919493926A3D7C6862FEC7F330152F400C0AECB4467508A"}}, }, { Query: "SELECT hex(v1), hex(v2) FROM hex_nums2;", Expected: []sql.Row{{"765A8CE4CE74B187", "148AA875C3CDB9AF8919493926A3D7C6862FEC7F330152F400C0AECB4467508A"}}, }, }, }, { Name: "Multialter DDL with ADD/DROP Primary Key", SetUpScript: []string{ "CREATE TABLE t(pk int primary key, v1 int)", }, Assertions: []ScriptTestAssertion{ { Query: "ALTER TABLE t ADD COLUMN (v2 int), drop primary key, add primary key (v2)", Expected: []sql.Row{{sql.NewOkResult(0)}}, }, { Query: "DESCRIBE t", Expected: []sql.Row{ {"pk", "int", "NO", "", "", ""}, {"v1", "int", "YES", "", "", ""}, {"v2", "int", "NO", "PRI", "", ""}, }, }, { Query: "ALTER TABLE t ADD COLUMN (v3 int), drop primary key, add primary key (notacolumn)", ExpectedErr: sql.ErrKeyColumnDoesNotExist, }, { Query: "DESCRIBE t", Expected: []sql.Row{ {"pk", "int", "NO", "", "", ""}, {"v1", "int", "YES", "", "", ""}, {"v2", "int", "NO", "PRI", "", ""}, }, }, }, }, { Name: "Multialter DDL with ADD/DROP INDEX", SetUpScript: []string{ "CREATE TABLE t(pk int primary key, v1 int)", }, Assertions: []ScriptTestAssertion{ { Query: "ALTER TABLE t DROP COLUMN v1, ADD INDEX myidx (v1)", ExpectedErr: sql.ErrKeyColumnDoesNotExist, }, { Query: "DESCRIBE t", Expected: []sql.Row{ {"pk", "int", "NO", "PRI", "", ""}, {"v1", "int", "YES", "", "", ""}, }, }, { Query: "ALTER TABLE t ADD COLUMN (v2 int), ADD INDEX myidx (v2)", Expected: []sql.Row{{sql.NewOkResult(0)}}, }, { Query: "DESCRIBE t", Expected: []sql.Row{ {"pk", "int", "NO", "PRI", "", ""}, {"v1", "int", "YES", "", "", ""}, {"v2", "int", "YES", "MUL", "", ""}, }, }, { Query: "ALTER TABLE t ADD COLUMN (v3 int), DROP INDEX notanindex", ExpectedErr: sql.ErrCantDropFieldOrKey, }, { Query: "DESCRIBE t", Expected: []sql.Row{ {"pk", "int", "NO", "PRI", "", ""}, {"v1", "int", "YES", "", "", ""}, {"v2", "int", "YES", "MUL", "", ""}, }, }, { Query: "ALTER TABLE t ADD COLUMN (v4 int), ADD INDEX myidx (notacolumn)", ExpectedErr: sql.ErrKeyColumnDoesNotExist, }, { Query: "DESCRIBE t", Expected: []sql.Row{ {"pk", "int", "NO", "PRI", "", ""}, {"v1", "int", "YES", "", "", ""}, {"v2", "int", "YES", "MUL", "", ""}, }, }, { Query: "ALTER TABLE t ADD COLUMN (v4 int), ADD INDEX myidx2 (v4), DROP INDEX notanindex;", ExpectedErr: sql.ErrCantDropFieldOrKey, }, { Query: "DESCRIBE t", Expected: []sql.Row{ {"pk", "int", "NO", "PRI", "", ""}, {"v1", "int", "YES", "", "", ""}, {"v2", "int", "YES", "MUL", "", ""}, }, }, { Query: "ALTER TABLE t ADD COLUMN (v4 int), ADD INDEX myidx2 (v4)", Expected: []sql.Row{{sql.NewOkResult(0)}}, }, { Query: "DESCRIBE t", Expected: []sql.Row{ {"pk", "int", "NO", "PRI", "", ""}, {"v1", "int", "YES", "", "", ""}, {"v2", "int", "YES", "MUL", "", ""}, {"v4", "int", "YES", "MUL", "", ""}, }, }, { Query: "ALTER TABLE t ADD COLUMN (v5 int), RENAME INDEX myidx2 TO myidx3", Expected: []sql.Row{{sql.NewOkResult(0)}}, }, { Query: "ALTER TABLE t DROP INDEX myidx, ADD INDEX v5idx (v5)", Expected: []sql.Row{{sql.NewOkResult(0)}}, }, { Query: "DESCRIBE t", Expected: []sql.Row{ {"pk", "int", "NO", "PRI", "", ""}, {"v1", "int", "YES", "", "", ""}, {"v2", "int", "YES", "", "", ""}, {"v4", "int", "YES", "MUL", "", ""}, {"v5", "int", "YES", "MUL", "", ""}, }, }, }, }, { Name: "ALTER TABLE MULTI ADD/DROP COLUMN", SetUpScript: []string{ "CREATE TABLE test (pk BIGINT PRIMARY KEY, v1 BIGINT NOT NULL DEFAULT 88);", }, Assertions: []ScriptTestAssertion{ { Query: "INSERT INTO test (pk) VALUES (1);", Expected: []sql.Row{{sql.NewOkResult(1)}}, }, { Query: "ALTER TABLE test DROP COLUMN v1, ADD COLUMN v2 INT NOT NULL DEFAULT 100", Expected: []sql.Row{{sql.NewOkResult(0)}}, }, { Query: "describe test", Expected: []sql.Row{ {"pk", "bigint", "NO", "PRI", "", ""}, {"v2", "int", "NO", "", "100", ""}, }, }, { Query: "ALTER TABLE TEST MODIFY COLUMN pk BIGINT AUTO_INCREMENT, AUTO_INCREMENT = 100", Expected: []sql.Row{{sql.NewOkResult(0)}}, }, { Query: "INSERT INTO test (v2) values (11)", Expected: []sql.Row{{sql.OkResult{RowsAffected: 1, InsertID: 100}}}, }, { Query: "SELECT * from test where pk = 100", Expected: []sql.Row{{100, 11}}, }, { Query: "ALTER TABLE test DROP COLUMN v2, ADD COLUMN v3 int NOT NULL after v2", ExpectedErr: sql.ErrTableColumnNotFound, }, { Query: "describe test", Expected: []sql.Row{ {"pk", "bigint", "NO", "PRI", "", "auto_increment"}, {"v2", "int", "NO", "", "100", ""}, }, }, { Query: "ALTER TABLE test DROP COLUMN v2, RENAME COLUMN v2 to v3", ExpectedErr: sql.ErrTableColumnNotFound, }, { Query: "describe test", Expected: []sql.Row{ {"pk", "bigint", "NO", "PRI", "", "auto_increment"}, {"v2", "int", "NO", "", "100", ""}, }, }, { Query: "ALTER TABLE test RENAME COLUMN v2 to v3, DROP COLUMN v2", ExpectedErr: sql.ErrTableColumnNotFound, }, { Query: "describe test", Expected: []sql.Row{ {"pk", "bigint", "NO", "PRI", "", "auto_increment"}, {"v2", "int", "NO", "", "100", ""}, }, }, { Query: "ALTER TABLE test ADD COLUMN (v3 int NOT NULL), add column (v4 int), drop column v2, add column (v5 int NOT NULL)", Expected: []sql.Row{{sql.NewOkResult(0)}}, }, { Query: "DESCRIBE test", Expected: []sql.Row{ {"pk", "bigint", "NO", "PRI", "", "auto_increment"}, {"v3", "int", "NO", "", "", ""}, {"v4", "int", "YES", "", "", ""}, {"v5", "int", "NO", "", "", ""}, }, }, { Query: "ALTER TABLE test ADD COLUMN (v6 int not null), RENAME COLUMN v5 TO mycol, DROP COLUMN v4, ADD COLUMN (v7 int);", Expected: []sql.Row{{sql.NewOkResult(0)}}, }, { Query: "describe test", Expected: []sql.Row{ {"pk", "bigint", "NO", "PRI", "", "auto_increment"}, {"v3", "int", "NO", "", "", ""}, {"mycol", "int", "NO", "", "", ""}, {"v6", "int", "NO", "", "", ""}, {"v7", "int", "YES", "", "", ""}, }, }, }, }, { Name: "join index lookups do not handle filters", SetUpScript: []string{ "create table a (x int primary key)", "create table b (y int primary key, x int, index idx_x(x))", "create table c (z int primary key, x int, y int, index idx_x(x))", "insert into a values (0),(1),(2),(3)", "insert into b values (0,1), (1,1), (2,2), (3,2)", "insert into c values (0,1,0), (1,1,0), (2,2,1), (3,2,1)", }, Query: "select a.* from a join b on a.x = b.x join c where c.x = a.x and b.x = 1", Expected: []sql.Row{ {1}, {1}, {1}, {1}, }, }, { Name: "failed conversion shows warning", Assertions: []ScriptTestAssertion{ { Query: "SELECT CONVERT('10000-12-31 23:59:59', DATETIME)", ExpectedWarning: 1292, ExpectedWarningsCount: 1, ExpectedWarningMessageSubstring: "Incorrect datetime value: 10000-12-31 23:59:59", SkipResultsCheck: true, }, { Query: "SELECT CONVERT('this is not a datetime', DATETIME)", ExpectedWarning: 1292, ExpectedWarningsCount: 1, ExpectedWarningMessageSubstring: "Incorrect datetime value: this is not a datetime", SkipResultsCheck: true, }, { Query: "SELECT CAST('this is not a datetime' as DATETIME)", ExpectedWarning: 1292, ExpectedWarningsCount: 1, ExpectedWarningMessageSubstring: "Incorrect datetime value: this is not a datetime", SkipResultsCheck: true, }, { Query: "SELECT CONVERT('this is not a date', DATE)", ExpectedWarning: 1292, ExpectedWarningsCount: 1, ExpectedWarningMessageSubstring: "Incorrect date value: this is not a date", SkipResultsCheck: true, }, { Query: "SELECT CAST('this is not a date' as DATE)", ExpectedWarning: 1292, ExpectedWarningsCount: 1, ExpectedWarningMessageSubstring: "Incorrect date value: this is not a date", SkipResultsCheck: true, }, }, }, }
ScriptTests are a set of test scripts to run. Unlike other engine tests, ScriptTests must be self-contained. No other tables are created outside the definition of the tests.
var ServerAuthTests = []ServerAuthenticationTest{ { Name: "Basic root authentication", Assertions: []ServerAuthenticationTestAssertion{ { Username: "root", Password: "", Query: "SELECT * FROM mysql.user;", ExpectedErr: false, }, { Username: "root", Password: "pass", Query: "SELECT * FROM mysql.user;", ExpectedErr: true, }, }, }, { Name: "Create User without plugin specification", SetUpScript: []string{ "CREATE USER rand_user@localhost IDENTIFIED BY 'rand_pass';", "GRANT ALL ON *.* TO rand_user@localhost WITH GRANT OPTION;", }, Assertions: []ServerAuthenticationTestAssertion{ { Username: "rand_user", Password: "rand_pass", Query: "SELECT * FROM mysql.user;", ExpectedErr: false, }, { Username: "rand_user", Password: "rand_pass1", Query: "SELECT * FROM mysql.user;", ExpectedErr: true, }, { Username: "rand_user", Password: "", Query: "SELECT * FROM mysql.user;", ExpectedErr: true, }, { Username: "rand_use", Password: "rand_pass", Query: "SELECT * FROM mysql.user;", ExpectedErr: true, }, }, }, { Name: "Create User with plugin specification", SetUpScript: []string{ "CREATE USER ranuse@localhost IDENTIFIED WITH mysql_native_password BY 'ranpas';", "GRANT ALL ON *.* TO ranuse@localhost WITH GRANT OPTION;", }, Assertions: []ServerAuthenticationTestAssertion{ { Username: "ranuse", Password: "ranpas", Query: "SELECT * FROM mysql.user;", ExpectedErr: false, }, { Username: "ranuse", Password: "what", Query: "SELECT * FROM mysql.user;", ExpectedErr: true, }, { Username: "ranuse", Password: "", Query: "SELECT * FROM mysql.user;", ExpectedErr: true, }, }, }, { Name: "Adding a Super User directly", SetUpFunc: func(ctx *sql.Context, t *testing.T, engine *sqle.Engine) { engine.Analyzer.Catalog.MySQLDb.AddSuperUser("bestuser", "the_pass") }, Assertions: []ServerAuthenticationTestAssertion{ { Username: "bestuser", Password: "the_past", Query: "SELECT * FROM mysql.user;", ExpectedErr: true, }, { Username: "bestuser", Password: "the_pass", Query: "SELECT * FROM mysql.user;", ExpectedErr: false, }, }, }, }
ServerAuthTests test the server authentication system. These tests always have the root account available, and the root account is used with any queries in the SetUpScript, along as being set to the context passed to SetUpFunc.
var ShowTableStatusQueries = []QueryTest{ { Query: `SHOW TABLE STATUS FROM mydb`, Expected: []sql.Row{ {"mytable", "InnoDB", "10", "Fixed", uint64(3), uint64(88), uint64(264), uint64(0), int64(0), int64(0), nil, nil, nil, nil, "utf8mb4_0900_bin", nil, nil, nil}, {"othertable", "InnoDB", "10", "Fixed", uint64(3), uint64(65540), uint64(196620), uint64(0), int64(0), int64(0), nil, nil, nil, nil, "utf8mb4_0900_bin", nil, nil, nil}, }, }, { Query: `SHOW TABLE STATUS LIKE '%table'`, Expected: []sql.Row{ {"mytable", "InnoDB", "10", "Fixed", uint64(3), uint64(88), uint64(264), uint64(0), int64(0), int64(0), nil, nil, nil, nil, "utf8mb4_0900_bin", nil, nil, nil}, {"othertable", "InnoDB", "10", "Fixed", uint64(3), uint64(65540), uint64(196620), uint64(0), int64(0), int64(0), nil, nil, nil, nil, "utf8mb4_0900_bin", nil, nil, nil}, }, }, { Query: `SHOW TABLE STATUS FROM mydb LIKE 'othertable'`, Expected: []sql.Row{ {"othertable", "InnoDB", "10", "Fixed", uint64(3), uint64(65540), uint64(196620), uint64(0), int64(0), int64(0), nil, nil, nil, nil, "utf8mb4_0900_bin", nil, nil, nil}, }, }, { Query: `SHOW TABLE STATUS WHERE Name = 'mytable'`, Expected: []sql.Row{ {"mytable", "InnoDB", "10", "Fixed", uint64(3), uint64(88), uint64(264), uint64(0), int64(0), int64(0), nil, nil, nil, nil, "utf8mb4_0900_bin", nil, nil, nil}, }, }, { Query: `SHOW TABLE STATUS`, Expected: []sql.Row{ {"mytable", "InnoDB", "10", "Fixed", uint64(3), uint64(88), uint64(264), uint64(0), int64(0), int64(0), nil, nil, nil, nil, "utf8mb4_0900_bin", nil, nil, nil}, {"othertable", "InnoDB", "10", "Fixed", uint64(3), uint64(65540), uint64(196620), uint64(0), int64(0), int64(0), nil, nil, nil, nil, "utf8mb4_0900_bin", nil, nil, nil}, }, }, { Query: `SHOW TABLE STATUS FROM mydb LIKE 'othertable'`, Expected: []sql.Row{ {"othertable", "InnoDB", "10", "Fixed", uint64(3), uint64(65540), uint64(196620), uint64(0), int64(0), int64(0), nil, nil, nil, nil, "utf8mb4_0900_bin", nil, nil, nil}, }, }, }
var SkippedJoinQueryTests = []QueryTest{ { Query: "select a.* from one_pk_two_idx a LEFT JOIN (one_pk_two_idx i JOIN one_pk_three_idx j on i.pk = j.v3) on a.pk = i.pk LEFT JOIN (one_pk_two_idx k JOIN one_pk_three_idx l on k.v2 = l.v3) on a.v1 = l.v2;", Expected: []sql.Row{{0, 0, 0}, {0, 0, 0}, {0, 0, 0}, {0, 0, 0}, {0, 0, 0}, {0, 0, 0}, {0, 0, 0}, {0, 0, 0}, {0, 0, 0}, {0, 0, 0}, {0, 0, 0}, {0, 0, 0}, {0, 0, 0}, {0, 0, 0}, {0, 0, 0}, {0, 0, 0}, {1, 1, 1}, {2, 2, 2}, {3, 3, 3}, {4, 4, 4}, {5, 5, 5}, {6, 6, 6}, {7, 7, 7}, }, }, }
var SkippedUpdateTests = []WriteQueryTest{ { WriteQuery: `UPDATE one_pk INNER JOIN two_pk on one_pk.pk = two_pk.pk1 SET one_pk.c1 = one_pk.c1 + 1, two_pk.c1 = two_pk.c2 + 1`, ExpectedWriteResult: []sql.Row{{newUpdateResult(8, 6)}}, SelectQuery: "SELECT * FROM two_pk;", ExpectedSelect: []sql.Row{ sql.NewRow(0, 0, 2, 1, 2, 3, 4), sql.NewRow(0, 1, 12, 11, 12, 13, 14), sql.NewRow(1, 0, 22, 21, 22, 23, 24), sql.NewRow(1, 1, 32, 31, 32, 33, 34), }, }, { WriteQuery: `UPDATE othertable INNER JOIN tabletest on othertable.i2=3 and tabletest.i=3 SET othertable.s2 = 'fourth'`, ExpectedWriteResult: []sql.Row{{newUpdateResult(1, 1)}}, SelectQuery: "SELECT * FROM othertable;", ExpectedSelect: []sql.Row{ sql.NewRow("third", 1), sql.NewRow("second", 2), sql.NewRow("fourth", 3), }, }, }
These tests return the correct select query answer but the wrong write result.
var SpatialDeleteTests = []WriteQueryTest{ { WriteQuery: "DELETE FROM point_table;", ExpectedWriteResult: []sql.Row{{sql.NewOkResult(1)}}, SelectQuery: "SELECT * FROM point_table;", ExpectedSelect: nil, }, { WriteQuery: "DELETE FROM line_table;", ExpectedWriteResult: []sql.Row{{sql.NewOkResult(2)}}, SelectQuery: "SELECT * FROM line_table;", ExpectedSelect: nil, }, { WriteQuery: "DELETE FROM polygon_table;", ExpectedWriteResult: []sql.Row{{sql.NewOkResult(1)}}, SelectQuery: "SELECT * FROM polygon_table;", ExpectedSelect: nil, }, }
var SpatialInsertQueries = []WriteQueryTest{ { WriteQuery: "INSERT INTO point_table VALUES (1, POINT(1,1));", ExpectedWriteResult: []sql.Row{{sql.NewOkResult(1)}}, SelectQuery: "SELECT * FROM point_table;", ExpectedSelect: []sql.Row{{5, sql.Point{X: 1, Y: 2}}, {1, sql.Point{X: 1, Y: 1}}}, }, { WriteQuery: "INSERT INTO point_table VALUES (1, 0x000000000101000000000000000000F03F0000000000000040);", ExpectedWriteResult: []sql.Row{{sql.NewOkResult(1)}}, SelectQuery: "SELECT * FROM point_table;", ExpectedSelect: []sql.Row{{5, sql.Point{X: 1, Y: 2}}, {1, sql.Point{X: 1, Y: 2}}}, }, { WriteQuery: "INSERT INTO line_table VALUES (2, LINESTRING(POINT(1,2),POINT(3,4)));", ExpectedWriteResult: []sql.Row{{sql.NewOkResult(1)}}, SelectQuery: "SELECT * FROM line_table;", ExpectedSelect: []sql.Row{{0, sql.LineString{Points: []sql.Point{{X: 1, Y: 2}, {X: 3, Y: 4}}}}, {1, sql.LineString{Points: []sql.Point{{X: 1, Y: 2}, {X: 3, Y: 4}, {X: 5, Y: 6}}}}, {2, sql.LineString{Points: []sql.Point{{X: 1, Y: 2}, {X: 3, Y: 4}}}}}, }, { WriteQuery: "INSERT INTO line_table VALUES (2, 0x00000000010200000002000000000000000000F03F000000000000004000000000000008400000000000001040);", ExpectedWriteResult: []sql.Row{{sql.NewOkResult(1)}}, SelectQuery: "SELECT * FROM line_table;", ExpectedSelect: []sql.Row{{0, sql.LineString{Points: []sql.Point{{X: 1, Y: 2}, {X: 3, Y: 4}}}}, {1, sql.LineString{Points: []sql.Point{{X: 1, Y: 2}, {X: 3, Y: 4}, {X: 5, Y: 6}}}}, {2, sql.LineString{Points: []sql.Point{{X: 1, Y: 2}, {X: 3, Y: 4}}}}}, }, { WriteQuery: "INSERT INTO polygon_table VALUES (1, POLYGON(LINESTRING(POINT(1,1),POINT(1,-1),POINT(-1,-1),POINT(-1,1),POINT(1,1))));", ExpectedWriteResult: []sql.Row{{sql.NewOkResult(1)}}, SelectQuery: "SELECT * FROM polygon_table;", ExpectedSelect: []sql.Row{{0, sql.Polygon{Lines: []sql.LineString{{Points: []sql.Point{{X: 0, Y: 0}, {X: 0, Y: 1}, {X: 1, Y: 1}, {X: 0, Y: 0}}}}}}, {1, sql.Polygon{Lines: []sql.LineString{{Points: []sql.Point{{X: 1, Y: 1}, {X: 1, Y: -1}, {X: -1, Y: -1}, {X: -1, Y: 1}, {X: 1, Y: 1}}}}}}}, }, { WriteQuery: "INSERT INTO polygon_table VALUES (1, 0x0000000001030000000100000005000000000000000000F03F000000000000F03F000000000000F03F000000000000F0BF000000000000F0BF000000000000F0BF000000000000F0BF000000000000F03F000000000000F03F000000000000F03F);", ExpectedWriteResult: []sql.Row{{sql.NewOkResult(1)}}, SelectQuery: "SELECT * FROM polygon_table;", ExpectedSelect: []sql.Row{{0, sql.Polygon{Lines: []sql.LineString{{Points: []sql.Point{{X: 0, Y: 0}, {X: 0, Y: 1}, {X: 1, Y: 1}, {X: 0, Y: 0}}}}}}, {1, sql.Polygon{Lines: []sql.LineString{{Points: []sql.Point{{X: 1, Y: 1}, {X: 1, Y: -1}, {X: -1, Y: -1}, {X: -1, Y: 1}, {X: 1, Y: 1}}}}}}}, }, { WriteQuery: "INSERT INTO geometry_table VALUES (7, POINT(123.456,7.89));", ExpectedWriteResult: []sql.Row{{sql.NewOkResult(1)}}, SelectQuery: "SELECT * FROM geometry_table;", ExpectedSelect: []sql.Row{ {1, sql.Point{X: 1, Y: 2}}, {2, sql.LineString{Points: []sql.Point{{X: 1, Y: 2}, {X: 3, Y: 4}}}}, {3, sql.Polygon{Lines: []sql.LineString{{Points: []sql.Point{{X: 0, Y: 0}, {X: 0, Y: 1}, {X: 1, Y: 1}, {X: 0, Y: 0}}}}}}, {4, sql.Point{SRID: 4326, X: 1, Y: 2}}, {5, sql.LineString{SRID: 4326, Points: []sql.Point{{SRID: 4326, X: 1, Y: 2}, {SRID: 4326, X: 3, Y: 4}}}}, {6, sql.Polygon{SRID: 4326, Lines: []sql.LineString{{SRID: 4326, Points: []sql.Point{{SRID: 4326, X: 0, Y: 0}, {SRID: 4326, X: 0, Y: 1}, {SRID: 4326, X: 1, Y: 1}, {SRID: 4326, X: 0, Y: 0}}}}}}, {7, sql.Point{X: 123.456, Y: 7.89}}, }, }, { WriteQuery: "INSERT INTO geometry_table VALUES (7, 0x00000000010100000077BE9F1A2FDD5E408FC2F5285C8F1F40);", ExpectedWriteResult: []sql.Row{{sql.NewOkResult(1)}}, SelectQuery: "SELECT * FROM geometry_table;", ExpectedSelect: []sql.Row{ {1, sql.Point{X: 1, Y: 2}}, {2, sql.LineString{Points: []sql.Point{{X: 1, Y: 2}, {X: 3, Y: 4}}}}, {3, sql.Polygon{Lines: []sql.LineString{{Points: []sql.Point{{X: 0, Y: 0}, {X: 0, Y: 1}, {X: 1, Y: 1}, {X: 0, Y: 0}}}}}}, {4, sql.Point{SRID: 4326, X: 1, Y: 2}}, {5, sql.LineString{SRID: 4326, Points: []sql.Point{{SRID: 4326, X: 1, Y: 2}, {SRID: 4326, X: 3, Y: 4}}}}, {6, sql.Polygon{SRID: 4326, Lines: []sql.LineString{{SRID: 4326, Points: []sql.Point{{SRID: 4326, X: 0, Y: 0}, {SRID: 4326, X: 0, Y: 1}, {SRID: 4326, X: 1, Y: 1}, {SRID: 4326, X: 0, Y: 0}}}}}}, {7, sql.Point{X: 123.456, Y: 7.89}}, }, }, { WriteQuery: "INSERT INTO geometry_table VALUES (7, LINESTRING(POINT(1,2),POINT(3,4)));", ExpectedWriteResult: []sql.Row{{sql.NewOkResult(1)}}, SelectQuery: "SELECT * FROM geometry_table;", ExpectedSelect: []sql.Row{ {1, sql.Point{X: 1, Y: 2}}, {2, sql.LineString{Points: []sql.Point{{X: 1, Y: 2}, {X: 3, Y: 4}}}}, {3, sql.Polygon{Lines: []sql.LineString{{Points: []sql.Point{{X: 0, Y: 0}, {X: 0, Y: 1}, {X: 1, Y: 1}, {X: 0, Y: 0}}}}}}, {4, sql.Point{SRID: 4326, X: 1, Y: 2}}, {5, sql.LineString{SRID: 4326, Points: []sql.Point{{SRID: 4326, X: 1, Y: 2}, {SRID: 4326, X: 3, Y: 4}}}}, {6, sql.Polygon{SRID: 4326, Lines: []sql.LineString{{SRID: 4326, Points: []sql.Point{{SRID: 4326, X: 0, Y: 0}, {SRID: 4326, X: 0, Y: 1}, {SRID: 4326, X: 1, Y: 1}, {SRID: 4326, X: 0, Y: 0}}}}}}, {7, sql.LineString{Points: []sql.Point{{X: 1, Y: 2}, {X: 3, Y: 4}}}}, }, }, { WriteQuery: "INSERT INTO geometry_table VALUES (7, 0x00000000010200000002000000000000000000F03F000000000000004000000000000008400000000000001040);", ExpectedWriteResult: []sql.Row{{sql.NewOkResult(1)}}, SelectQuery: "SELECT * FROM geometry_table;", ExpectedSelect: []sql.Row{ {1, sql.Point{X: 1, Y: 2}}, {2, sql.LineString{Points: []sql.Point{{X: 1, Y: 2}, {X: 3, Y: 4}}}}, {3, sql.Polygon{Lines: []sql.LineString{{Points: []sql.Point{{X: 0, Y: 0}, {X: 0, Y: 1}, {X: 1, Y: 1}, {X: 0, Y: 0}}}}}}, {4, sql.Point{SRID: 4326, X: 1, Y: 2}}, {5, sql.LineString{SRID: 4326, Points: []sql.Point{{SRID: 4326, X: 1, Y: 2}, {SRID: 4326, X: 3, Y: 4}}}}, {6, sql.Polygon{SRID: 4326, Lines: []sql.LineString{{SRID: 4326, Points: []sql.Point{{SRID: 4326, X: 0, Y: 0}, {SRID: 4326, X: 0, Y: 1}, {SRID: 4326, X: 1, Y: 1}, {SRID: 4326, X: 0, Y: 0}}}}}}, {7, sql.LineString{Points: []sql.Point{{X: 1, Y: 2}, {X: 3, Y: 4}}}}, }, }, { WriteQuery: "INSERT INTO geometry_table VALUES (7, POLYGON(LINESTRING(POINT(1,1),POINT(1,-1),POINT(-1,-1),POINT(-1,1),POINT(1,1))));", ExpectedWriteResult: []sql.Row{{sql.NewOkResult(1)}}, SelectQuery: "SELECT * FROM geometry_table;", ExpectedSelect: []sql.Row{ {1, sql.Point{X: 1, Y: 2}}, {2, sql.LineString{Points: []sql.Point{{X: 1, Y: 2}, {X: 3, Y: 4}}}}, {3, sql.Polygon{Lines: []sql.LineString{{Points: []sql.Point{{X: 0, Y: 0}, {X: 0, Y: 1}, {X: 1, Y: 1}, {X: 0, Y: 0}}}}}}, {4, sql.Point{SRID: 4326, X: 1, Y: 2}}, {5, sql.LineString{SRID: 4326, Points: []sql.Point{{SRID: 4326, X: 1, Y: 2}, {SRID: 4326, X: 3, Y: 4}}}}, {6, sql.Polygon{SRID: 4326, Lines: []sql.LineString{{SRID: 4326, Points: []sql.Point{{SRID: 4326, X: 0, Y: 0}, {SRID: 4326, X: 0, Y: 1}, {SRID: 4326, X: 1, Y: 1}, {SRID: 4326, X: 0, Y: 0}}}}}}, {7, sql.Polygon{Lines: []sql.LineString{{Points: []sql.Point{{X: 1, Y: 1}, {X: 1, Y: -1}, {X: -1, Y: -1}, {X: -1, Y: 1}, {X: 1, Y: 1}}}}}}, }, }, { WriteQuery: "INSERT INTO geometry_table VALUES (7, 0x0000000001030000000100000005000000000000000000F03F000000000000F03F000000000000F03F000000000000F0BF000000000000F0BF000000000000F0BF000000000000F0BF000000000000F03F000000000000F03F000000000000F03F);", ExpectedWriteResult: []sql.Row{{sql.NewOkResult(1)}}, SelectQuery: "SELECT * FROM geometry_table;", ExpectedSelect: []sql.Row{ {1, sql.Point{X: 1, Y: 2}}, {2, sql.LineString{Points: []sql.Point{{X: 1, Y: 2}, {X: 3, Y: 4}}}}, {3, sql.Polygon{Lines: []sql.LineString{{Points: []sql.Point{{X: 0, Y: 0}, {X: 0, Y: 1}, {X: 1, Y: 1}, {X: 0, Y: 0}}}}}}, {4, sql.Point{SRID: 4326, X: 1, Y: 2}}, {5, sql.LineString{SRID: 4326, Points: []sql.Point{{SRID: 4326, X: 1, Y: 2}, {SRID: 4326, X: 3, Y: 4}}}}, {6, sql.Polygon{SRID: 4326, Lines: []sql.LineString{{SRID: 4326, Points: []sql.Point{{SRID: 4326, X: 0, Y: 0}, {SRID: 4326, X: 0, Y: 1}, {SRID: 4326, X: 1, Y: 1}, {SRID: 4326, X: 0, Y: 0}}}}}}, {7, sql.Polygon{Lines: []sql.LineString{{Points: []sql.Point{{X: 1, Y: 1}, {X: 1, Y: -1}, {X: -1, Y: -1}, {X: -1, Y: 1}, {X: 1, Y: 1}}}}}}, }, }, }
var SpatialQueryTests = []QueryTest{ { Query: `SHOW CREATE TABLE point_table`, Expected: []sql.Row{{ "point_table", "CREATE TABLE `point_table` (\n" + " `i` bigint NOT NULL,\n" + " `p` point NOT NULL,\n" + " PRIMARY KEY (`i`)\n" + ") ENGINE=InnoDB DEFAULT CHARSET=utf8mb4 COLLATE=utf8mb4_0900_bin", }}, }, { Query: `SHOW CREATE TABLE line_table`, Expected: []sql.Row{{ "line_table", "CREATE TABLE `line_table` (\n" + " `i` bigint NOT NULL,\n" + " `l` linestring NOT NULL,\n" + " PRIMARY KEY (`i`)\n" + ") ENGINE=InnoDB DEFAULT CHARSET=utf8mb4 COLLATE=utf8mb4_0900_bin", }}, }, { Query: `SHOW CREATE TABLE polygon_table`, Expected: []sql.Row{{ "polygon_table", "CREATE TABLE `polygon_table` (\n" + " `i` bigint NOT NULL,\n" + " `p` polygon NOT NULL,\n" + " PRIMARY KEY (`i`)\n" + ") ENGINE=InnoDB DEFAULT CHARSET=utf8mb4 COLLATE=utf8mb4_0900_bin", }}, }, { Query: `SHOW CREATE TABLE geometry_table`, Expected: []sql.Row{{ "geometry_table", "CREATE TABLE `geometry_table` (\n" + " `i` bigint NOT NULL,\n" + " `g` geometry NOT NULL,\n" + " PRIMARY KEY (`i`)\n" + ") ENGINE=InnoDB DEFAULT CHARSET=utf8mb4 COLLATE=utf8mb4_0900_bin", }}, }, { Query: `SELECT HEX(ST_ASWKB(p)) from point_table`, Expected: []sql.Row{{"0101000000000000000000F03F0000000000000040"}}, }, { Query: `SELECT HEX(ST_ASWKB(l)) from line_table`, Expected: []sql.Row{ {"010200000002000000000000000000F03F000000000000004000000000000008400000000000001040"}, {"010200000003000000000000000000F03F00000000000000400000000000000840000000000000104000000000000014400000000000001840"}, }, }, { Query: `SELECT HEX(ST_ASWKB(p)) from polygon_table`, Expected: []sql.Row{{"01030000000100000004000000000000000000000000000000000000000000000000000000000000000000F03F000000000000F03F000000000000F03F00000000000000000000000000000000"}}, }, { Query: `SELECT ST_GEOMFROMWKB(ST_ASWKB(POINT(123.45,6.78)))`, Expected: []sql.Row{{sql.Point{X: 123.45, Y: 6.78}}}, }, { Query: `SELECT ST_GEOMFROMWKB(ST_ASWKB(LINESTRING(POINT(1.2,3.45),point(67.8,9))))`, Expected: []sql.Row{{sql.LineString{Points: []sql.Point{{X: 1.2, Y: 3.45}, {X: 67.8, Y: 9}}}}}, }, { Query: `SELECT ST_GEOMFROMWKB(ST_ASWKB(POLYGON(LINESTRING(POINT(0,0),POINT(2,2),POINT(1,1),POINT(0,0)))))`, Expected: []sql.Row{{sql.Polygon{Lines: []sql.LineString{{Points: []sql.Point{{X: 0, Y: 0}, {X: 2, Y: 2}, {X: 1, Y: 1}, {X: 0, Y: 0}}}}}}}, }, { Query: `SELECT ST_ASWKT(p) from point_table`, Expected: []sql.Row{{"POINT(1 2)"}}, }, { Query: `SELECT ST_ASWKT(l) from line_table`, Expected: []sql.Row{ {"LINESTRING(1 2,3 4)"}, {"LINESTRING(1 2,3 4,5 6)"}, }, }, { Query: `SELECT ST_ASWKT(p) from polygon_table`, Expected: []sql.Row{{"POLYGON((0 0,0 1,1 1,0 0))"}}, }, { Query: `SELECT ST_ASTEXT(p) from polygon_table`, Expected: []sql.Row{{"POLYGON((0 0,0 1,1 1,0 0))"}}, }, { Query: `SELECT ST_GEOMFROMTEXT(ST_ASWKT(POINT(1,2)))`, Expected: []sql.Row{{sql.Point{X: 1, Y: 2}}}, }, { Query: `SELECT ST_GEOMFROMTEXT(ST_ASWKT(LINESTRING(POINT(1.1,2.22),POINT(3.333,4.4444))))`, Expected: []sql.Row{{sql.LineString{Points: []sql.Point{{X: 1.1, Y: 2.22}, {X: 3.333, Y: 4.4444}}}}}, }, { Query: `SELECT ST_GEOMFROMTEXT(ST_ASWKT(POLYGON(LINESTRING(POINT(1.2, 3.4),POINT(2.5, -6.7),POINT(33, 44),POINT(1.2,3.4)))))`, Expected: []sql.Row{{sql.Polygon{Lines: []sql.LineString{{Points: []sql.Point{{X: 1.2, Y: 3.4}, {X: 2.5, Y: -6.7}, {X: 33, Y: 44}, {X: 1.2, Y: 3.4}}}}}}}, }, { Query: `SELECT ST_X(POINT(1,2))`, Expected: []sql.Row{{1.0}}, }, { Query: `SELECT ST_Y(POINT(1,2))`, Expected: []sql.Row{{2.0}}, }, { Query: `SELECT ST_X(POINT(123.45,6.789))`, Expected: []sql.Row{{123.45}}, }, { Query: `SELECT ST_Y(POINT(123.45,6.789))`, Expected: []sql.Row{{6.789}}, }, { Query: `SELECT ST_X(POINT(1,2),99.9)`, Expected: []sql.Row{{sql.Point{X: 99.9, Y: 2}}}, }, { Query: `SELECT ST_Y(POINT(1,2),99.9)`, Expected: []sql.Row{{sql.Point{X: 1, Y: 99.9}}}, }, { Query: `SELECT ST_X(p) from point_table`, Expected: []sql.Row{{1.0}}, }, { Query: `SELECT ST_X(p) from point_table`, Expected: []sql.Row{{1.0}}, }, { Query: `SELECT ST_Y(p) from point_table`, Expected: []sql.Row{{2.0}}, }, { Query: `SELECT ST_SRID(p) from point_table`, Expected: []sql.Row{{uint32(0)}}, }, { Query: `SELECT ST_SRID(l) from line_table`, Expected: []sql.Row{{uint32(0)}, {uint32(0)}}, }, { Query: `SELECT ST_SRID(p) from polygon_table`, Expected: []sql.Row{{uint32(0)}}, }, { Query: `SELECT ST_SRID(p, 4326) from point_table`, Expected: []sql.Row{{sql.Point{SRID: 4326, X: 1, Y: 2}}}, }, { Query: `SELECT ST_SRID(l, 4326) from line_table ORDER BY l`, Expected: []sql.Row{ {sql.LineString{SRID: 4326, Points: []sql.Point{{SRID: 4326, X: 1, Y: 2}, {SRID: 4326, X: 3, Y: 4}}}}, {sql.LineString{SRID: 4326, Points: []sql.Point{{SRID: 4326, X: 1, Y: 2}, {SRID: 4326, X: 3, Y: 4}, {SRID: 4326, X: 5, Y: 6}}}}, }, }, { Query: `SELECT ST_SRID(p, 4326) from polygon_table`, Expected: []sql.Row{ {sql.Polygon{SRID: 4326, Lines: []sql.LineString{{SRID: 4326, Points: []sql.Point{{SRID: 4326, X: 0, Y: 0}, {SRID: 4326, X: 0, Y: 1}, {SRID: 4326, X: 1, Y: 1}, {SRID: 4326, X: 0, Y: 0}}}}}}, }, }, { Query: `SELECT ST_GEOMFROMGEOJSON(s) from stringtogeojson_table`, Expected: []sql.Row{ {sql.Point{SRID: 4326, X: 2, Y: 1}}, {sql.Point{SRID: 4326, X: 56.789, Y: 123.45}}, {sql.LineString{SRID: 4326, Points: []sql.Point{{SRID: 4326, X: 2, Y: 1}, {SRID: 4326, X: 4, Y: 3}}}}, {sql.LineString{SRID: 4326, Points: []sql.Point{{SRID: 4326, X: 2.345, Y: 1.23}, {SRID: 4326, X: 4.56, Y: 3.56789}}}}, {sql.Polygon{SRID: 4326, Lines: []sql.LineString{{SRID: 4326, Points: []sql.Point{{SRID: 4326, X: 2.2, Y: 1.1}, {SRID: 4326, X: 4.4, Y: 3.3}, {SRID: 4326, X: 6.6, Y: 5.5}, {SRID: 4326, X: 2.2, Y: 1.1}}}}}}, {sql.Polygon{SRID: 4326, Lines: []sql.LineString{{SRID: 4326, Points: []sql.Point{{SRID: 4326, X: 0, Y: 0}, {SRID: 4326, X: 1, Y: 1}, {SRID: 4326, X: 2, Y: 2}, {SRID: 4326, X: 0, Y: 0}}}}}}, }, }, { Query: `SELECT ST_ASGEOJSON(p) from point_table`, Expected: []sql.Row{ {sql.JSONDocument{Val: map[string]interface{}{"type": "Point", "coordinates": [2]float64{1, 2}}}}, }, }, { Query: `SELECT ST_ASGEOJSON(l) from line_table`, Expected: []sql.Row{ {sql.JSONDocument{Val: map[string]interface{}{"type": "LineString", "coordinates": [][2]float64{{1, 2}, {3, 4}}}}}, {sql.JSONDocument{Val: map[string]interface{}{"type": "LineString", "coordinates": [][2]float64{{1, 2}, {3, 4}, {5, 6}}}}}, }, }, { Query: `SELECT ST_ASGEOJSON(p) from polygon_table`, Expected: []sql.Row{ {sql.JSONDocument{Val: map[string]interface{}{"type": "Polygon", "coordinates": [][][2]float64{{{0, 0}, {0, 1}, {1, 1}, {0, 0}}}}}}, }, }, { Query: `SELECT ST_ASGEOJSON(ST_GEOMFROMGEOJSON(s)) from stringtogeojson_table`, Expected: []sql.Row{ {sql.JSONDocument{Val: map[string]interface{}{"type": "Point", "coordinates": [2]float64{2, 1}}}}, {sql.JSONDocument{Val: map[string]interface{}{"type": "Point", "coordinates": [2]float64{56.789, 123.45}}}}, {sql.JSONDocument{Val: map[string]interface{}{"type": "LineString", "coordinates": [][2]float64{{2, 1}, {4, 3}}}}}, {sql.JSONDocument{Val: map[string]interface{}{"type": "LineString", "coordinates": [][2]float64{{2.345, 1.23}, {4.56, 3.56789}}}}}, {sql.JSONDocument{Val: map[string]interface{}{"type": "Polygon", "coordinates": [][][2]float64{{{2.2, 1.1}, {4.4, 3.3}, {6.6, 5.5}, {2.2, 1.1}}}}}}, {sql.JSONDocument{Val: map[string]interface{}{"type": "Polygon", "coordinates": [][][2]float64{{{0, 0}, {1, 1}, {2, 2}, {0, 0}}}}}}, }, }, { Query: `SELECT ST_GEOMFROMGEOJSON(ST_ASGEOJSON(p)) from point_table`, Expected: []sql.Row{ {sql.Point{SRID: 4326, X: 2, Y: 1}}, }, }, { Query: `SELECT ST_GEOMFROMGEOJSON(ST_ASGEOJSON(l)) from line_table`, Expected: []sql.Row{ {sql.LineString{SRID: 4326, Points: []sql.Point{{SRID: 4326, X: 2, Y: 1}, {SRID: 4326, X: 4, Y: 3}}}}, {sql.LineString{SRID: 4326, Points: []sql.Point{{SRID: 4326, X: 2, Y: 1}, {SRID: 4326, X: 4, Y: 3}, {SRID: 4326, X: 6, Y: 5}}}}, }, }, { Query: `SELECT ST_GEOMFROMGEOJSON(ST_ASGEOJSON(p)) from polygon_table`, Expected: []sql.Row{ {sql.Polygon{SRID: 4326, Lines: []sql.LineString{{SRID: 4326, Points: []sql.Point{{SRID: 4326, X: 0, Y: 0}, {SRID: 4326, X: 1, Y: 0}, {SRID: 4326, X: 1, Y: 1}, {SRID: 4326, X: 0, Y: 0}}}}}}, }, }, { Query: `SELECT ST_DIMENSION(p) from point_table`, Expected: []sql.Row{ {0}, }, }, { Query: `SELECT ST_DIMENSION(l) from line_table`, Expected: []sql.Row{ {1}, {1}, }, }, { Query: `SELECT ST_DIMENSION(p) from polygon_table`, Expected: []sql.Row{ {2}, }, }, { Query: `SELECT ST_SWAPXY(p) from point_table`, Expected: []sql.Row{ {sql.Point{X: 2, Y: 1}}, }, }, { Query: `SELECT ST_SWAPXY(l) from line_table`, Expected: []sql.Row{ {sql.LineString{Points: []sql.Point{{X: 2, Y: 1}, {X: 4, Y: 3}}}}, {sql.LineString{Points: []sql.Point{{X: 2, Y: 1}, {X: 4, Y: 3}, {X: 6, Y: 5}}}}, }, }, { Query: `SELECT ST_SWAPXY(p) from polygon_table`, Expected: []sql.Row{ {sql.Polygon{Lines: []sql.LineString{{Points: []sql.Point{{X: 0, Y: 0}, {X: 1, Y: 0}, {X: 1, Y: 1}, {X: 0, Y: 0}}}}}}, }, }, { Query: `SELECT ST_ASWKT(g) from geometry_table ORDER BY i`, Expected: []sql.Row{ {"POINT(1 2)"}, {"LINESTRING(1 2,3 4)"}, {"POLYGON((0 0,0 1,1 1,0 0))"}, {"POINT(2 1)"}, {"LINESTRING(2 1,4 3)"}, {"POLYGON((0 0,1 0,1 1,0 0))"}, }, }, { Query: `SELECT HEX(ST_ASWKB(g)) from geometry_table`, Expected: []sql.Row{ {"0101000000000000000000F03F0000000000000040"}, {"010200000002000000000000000000F03F000000000000004000000000000008400000000000001040"}, {"01030000000100000004000000000000000000000000000000000000000000000000000000000000000000F03F000000000000F03F000000000000F03F00000000000000000000000000000000"}, {"0101000000000000000000F03F0000000000000040"}, {"010200000002000000000000000000F03F000000000000004000000000000008400000000000001040"}, {"01030000000100000004000000000000000000000000000000000000000000000000000000000000000000F03F000000000000F03F000000000000F03F00000000000000000000000000000000"}, }, }, { Query: `SELECT ST_SRID(g) from geometry_table order by i`, Expected: []sql.Row{ {uint64(0)}, {uint64(0)}, {uint64(0)}, {uint64(4326)}, {uint64(4326)}, {uint64(4326)}, }, }, { Query: `SELECT ST_SRID(g, 0) from geometry_table order by i`, Expected: []sql.Row{ {sql.Point{X: 1, Y: 2}}, {sql.LineString{Points: []sql.Point{{X: 1, Y: 2}, {X: 3, Y: 4}}}}, {sql.Polygon{Lines: []sql.LineString{{Points: []sql.Point{{X: 0, Y: 0}, {X: 0, Y: 1}, {X: 1, Y: 1}, {X: 0, Y: 0}}}}}}, {sql.Point{X: 1, Y: 2}}, {sql.LineString{Points: []sql.Point{{X: 1, Y: 2}, {X: 3, Y: 4}}}}, {sql.Polygon{Lines: []sql.LineString{{Points: []sql.Point{{X: 0, Y: 0}, {X: 0, Y: 1}, {X: 1, Y: 1}, {X: 0, Y: 0}}}}}}, }, }, { Query: `SELECT ST_DIMENSION(g) from geometry_table order by i`, Expected: []sql.Row{ {0}, {1}, {2}, {0}, {1}, {2}, }, }, { Query: `SELECT ST_SWAPXY(g) from geometry_table order by i`, Expected: []sql.Row{ {sql.Point{X: 2, Y: 1}}, {sql.LineString{Points: []sql.Point{{X: 2, Y: 1}, {X: 4, Y: 3}}}}, {sql.Polygon{Lines: []sql.LineString{{Points: []sql.Point{{X: 0, Y: 0}, {X: 1, Y: 0}, {X: 1, Y: 1}, {X: 0, Y: 0}}}}}}, {sql.Point{SRID: 4326, X: 2, Y: 1}}, {sql.LineString{SRID: 4326, Points: []sql.Point{{SRID: 4326, X: 2, Y: 1}, {SRID: 4326, X: 4, Y: 3}}}}, {sql.Polygon{SRID: 4326, Lines: []sql.LineString{{SRID: 4326, Points: []sql.Point{{SRID: 4326, X: 0, Y: 0}, {SRID: 4326, X: 1, Y: 0}, {SRID: 4326, X: 1, Y: 1}, {SRID: 4326, X: 0, Y: 0}}}}}}, }, }, }
var SpatialScriptTests = []ScriptTest{ { Name: "create table using default point value", SetUpScript: []string{ "CREATE TABLE test (i int primary key, p point default point(123.456, 7.89));", "insert into test (i) values (0);", }, Assertions: []ScriptTestAssertion{ { Query: "select st_aswkt(p) from test", Expected: []sql.Row{{"POINT(123.456 7.89)"}}, }, { Query: "show create table test", Expected: []sql.Row{{"test", "CREATE TABLE `test` (\n `i` int NOT NULL,\n `p` point DEFAULT (POINT(123.456, 7.89)),\n PRIMARY KEY (`i`)\n) ENGINE=InnoDB DEFAULT CHARSET=utf8mb4 COLLATE=utf8mb4_0900_bin"}}, }, { Query: "describe test", Expected: []sql.Row{{"i", "int", "NO", "PRI", "", ""}, {"p", "point", "YES", "", "(POINT(123.456, 7.89))", ""}}, }, }, }, { Name: "create table using default linestring value", SetUpScript: []string{ "CREATE TABLE test (i int primary key, l linestring default linestring(point(1,2), point(3,4)));", "insert into test (i) values (0);", }, Assertions: []ScriptTestAssertion{ { Query: "select st_aswkt(l) from test", Expected: []sql.Row{{"LINESTRING(1 2,3 4)"}}, }, { Query: "show create table test", Expected: []sql.Row{{"test", "CREATE TABLE `test` (\n `i` int NOT NULL,\n `l` linestring DEFAULT (LINESTRING(POINT(1, 2),POINT(3, 4))),\n PRIMARY KEY (`i`)\n) ENGINE=InnoDB DEFAULT CHARSET=utf8mb4 COLLATE=utf8mb4_0900_bin"}}, }, { Query: "describe test", Expected: []sql.Row{{"i", "int", "NO", "PRI", "", ""}, {"l", "linestring", "YES", "", "(LINESTRING(POINT(1, 2),POINT(3, 4)))", ""}}, }, }, }, { Name: "create table using default polygon value", SetUpScript: []string{ "CREATE TABLE test (i int primary key, p polygon default polygon(linestring(point(0,0), point(1,1), point(2,2), point(0,0))));", "insert into test (i) values (0);", }, Assertions: []ScriptTestAssertion{ { Query: "select st_aswkt(p) from test", Expected: []sql.Row{{"POLYGON((0 0,1 1,2 2,0 0))"}}, }, { Query: "show create table test", Expected: []sql.Row{{"test", "CREATE TABLE `test` (\n `i` int NOT NULL,\n `p` polygon DEFAULT (POLYGON(LINESTRING(POINT(0, 0),POINT(1, 1),POINT(2, 2),POINT(0, 0)))),\n PRIMARY KEY (`i`)\n) ENGINE=InnoDB DEFAULT CHARSET=utf8mb4 COLLATE=utf8mb4_0900_bin"}}, }, { Query: "describe test", Expected: []sql.Row{{"i", "int", "NO", "PRI", "", ""}, {"p", "polygon", "YES", "", "(POLYGON(LINESTRING(POINT(0, 0),POINT(1, 1),POINT(2, 2),POINT(0, 0))))", ""}}, }, }, }, { Name: "create geometry table using default point value", SetUpScript: []string{ "CREATE TABLE test (i int primary key, g geometry default point(123.456, 7.89));", "insert into test (i) values (0);", }, Assertions: []ScriptTestAssertion{ { Query: "select st_aswkt(g) from test", Expected: []sql.Row{{"POINT(123.456 7.89)"}}, }, { Query: "show create table test", Expected: []sql.Row{{"test", "CREATE TABLE `test` (\n `i` int NOT NULL,\n `g` geometry DEFAULT (POINT(123.456, 7.89)),\n PRIMARY KEY (`i`)\n) ENGINE=InnoDB DEFAULT CHARSET=utf8mb4 COLLATE=utf8mb4_0900_bin"}}, }, { Query: "describe test", Expected: []sql.Row{{"i", "int", "NO", "PRI", "", ""}, {"g", "geometry", "YES", "", "(POINT(123.456, 7.89))", ""}}, }, }, }, { Name: "create geometry table using default linestring value", SetUpScript: []string{ "CREATE TABLE test (i int primary key, g geometry default linestring(point(1,2), point(3,4)));", "insert into test (i) values (0);", }, Assertions: []ScriptTestAssertion{ { Query: "select st_aswkt(g) from test", Expected: []sql.Row{{"LINESTRING(1 2,3 4)"}}, }, { Query: "show create table test", Expected: []sql.Row{{"test", "CREATE TABLE `test` (\n `i` int NOT NULL,\n `g` geometry DEFAULT (LINESTRING(POINT(1, 2),POINT(3, 4))),\n PRIMARY KEY (`i`)\n) ENGINE=InnoDB DEFAULT CHARSET=utf8mb4 COLLATE=utf8mb4_0900_bin"}}, }, { Query: "describe test", Expected: []sql.Row{{"i", "int", "NO", "PRI", "", ""}, {"g", "geometry", "YES", "", "(LINESTRING(POINT(1, 2),POINT(3, 4)))", ""}}, }, }, }, { Name: "create geometry table using default polygon value", SetUpScript: []string{ "CREATE TABLE test (i int primary key, g geometry default polygon(linestring(point(0,0), point(1,1), point(2,2), point(0,0))));", "insert into test (i) values (0);", }, Assertions: []ScriptTestAssertion{ { Query: "select st_aswkt(g) from test", Expected: []sql.Row{{"POLYGON((0 0,1 1,2 2,0 0))"}}, }, { Query: "show create table test", Expected: []sql.Row{{"test", "CREATE TABLE `test` (\n `i` int NOT NULL,\n `g` geometry DEFAULT (POLYGON(LINESTRING(POINT(0, 0),POINT(1, 1),POINT(2, 2),POINT(0, 0)))),\n PRIMARY KEY (`i`)\n) ENGINE=InnoDB DEFAULT CHARSET=utf8mb4 COLLATE=utf8mb4_0900_bin"}}, }, { Query: "describe test", Expected: []sql.Row{{"i", "int", "NO", "PRI", "", ""}, {"g", "geometry", "YES", "", "(POLYGON(LINESTRING(POINT(0, 0),POINT(1, 1),POINT(2, 2),POINT(0, 0))))", ""}}, }, }, }, { Name: "create table with NULL default values for geometry types", SetUpScript: []string{ "CREATE TABLE null_default (pk int NOT NULL PRIMARY KEY, v1 geometry DEFAULT NULL, v2 linestring DEFAULT NULL, v3 point DEFAULT NULL, v4 polygon DEFAULT NULL)", "insert into null_default(pk) values (0)", }, Assertions: []ScriptTestAssertion{ { Query: "select * from null_default", Expected: []sql.Row{{0, nil, nil, nil, nil}}, }, }, }, { Name: "create table using SRID value for geometry type", SetUpScript: []string{ "CREATE TABLE tab0 (i int primary key, g geometry srid 4326 default (point(1,1)));", }, Assertions: []ScriptTestAssertion{ { Query: "show create table tab0", Expected: []sql.Row{{"tab0", "CREATE TABLE `tab0` (\n `i` int NOT NULL,\n `g` geometry SRID 4326 DEFAULT (POINT(1, 1)),\n PRIMARY KEY (`i`)\n) ENGINE=InnoDB DEFAULT CHARSET=utf8mb4 COLLATE=utf8mb4_0900_bin"}}, }, { Query: "INSERT INTO tab0 VALUES (1, ST_GEOMFROMTEXT(ST_ASWKT(POINT(1,2)), 4326))", Expected: []sql.Row{{sql.NewOkResult(1)}}, }, { Query: "select i, ST_ASWKT(g) FROM tab0", Expected: []sql.Row{{1, "POINT(1 2)"}}, }, { Query: "INSERT INTO tab0 VALUES (2, ST_GEOMFROMTEXT(ST_ASWKT(POINT(2,4))))", ExpectedErr: sql.ErrNotMatchingSRIDWithColName, }, { Query: "INSERT INTO tab0 VALUES (2, ST_GEOMFROMTEXT(ST_ASWKT(LINESTRING(POINT(1, 6),POINT(4, 3))), 4326))", Expected: []sql.Row{{sql.NewOkResult(1)}}, }, { Query: "select i, ST_ASWKT(g) FROM tab0", Expected: []sql.Row{{1, "POINT(1 2)"}, {2, "LINESTRING(1 6,4 3)"}}, }, }, }, { Name: "create table using SRID value for linestring type", SetUpScript: []string{ "CREATE TABLE tab1 (i int primary key, l linestring srid 0);", }, Assertions: []ScriptTestAssertion{ { Query: "show create table tab1", Expected: []sql.Row{{"tab1", "CREATE TABLE `tab1` (\n `i` int NOT NULL,\n `l` linestring SRID 0,\n PRIMARY KEY (`i`)\n) ENGINE=InnoDB DEFAULT CHARSET=utf8mb4 COLLATE=utf8mb4_0900_bin"}}, }, { Query: "INSERT INTO tab1 VALUES (1, LINESTRING(POINT(0, 0),POINT(2, 2)))", Expected: []sql.Row{{sql.NewOkResult(1)}}, }, { Query: "select i, ST_ASWKT(l) FROM tab1", Expected: []sql.Row{{1, "LINESTRING(0 0,2 2)"}}, }, { Query: "INSERT INTO tab1 VALUES (2, ST_GEOMFROMTEXT(ST_ASWKT(LINESTRING(POINT(1, 6),POINT(4, 3))), 4326))", ExpectedErr: sql.ErrNotMatchingSRIDWithColName, }, { Query: "select i, ST_ASWKT(l) FROM tab1", Expected: []sql.Row{{1, "LINESTRING(0 0,2 2)"}}, }, }, }, { Name: "create table using SRID value for point type", SetUpScript: []string{ "CREATE TABLE tab2 (i int primary key);", }, Assertions: []ScriptTestAssertion{ { Query: "ALTER TABLE tab2 ADD COLUMN p POINT NOT NULL SRID 0", Expected: []sql.Row{{sql.NewOkResult(0)}}, }, { Query: "show create table tab2", Expected: []sql.Row{{"tab2", "CREATE TABLE `tab2` (\n `i` int NOT NULL,\n `p` point NOT NULL SRID 0,\n PRIMARY KEY (`i`)\n) ENGINE=InnoDB DEFAULT CHARSET=utf8mb4 COLLATE=utf8mb4_0900_bin"}}, }, { Query: "INSERT INTO tab2 VALUES (1, POINT(2, 2))", Expected: []sql.Row{{sql.NewOkResult(1)}}, }, { Query: "select i, ST_ASWKT(p) FROM tab2", Expected: []sql.Row{{1, "POINT(2 2)"}}, }, { Query: "INSERT INTO tab2 VALUES (2, ST_GEOMFROMTEXT(ST_ASWKT(POINT(1, 6)), 4326))", ExpectedErr: sql.ErrNotMatchingSRIDWithColName, }, { Query: "select i, ST_ASWKT(p) FROM tab2", Expected: []sql.Row{{1, "POINT(2 2)"}}, }, { Query: "ALTER TABLE tab2 CHANGE COLUMN p p POINT NOT NULL", Expected: []sql.Row{{sql.NewOkResult(0)}}, }, { Query: "INSERT INTO tab2 VALUES (2, ST_GEOMFROMTEXT(ST_ASWKT(POINT(1, 6)), 4326))", Expected: []sql.Row{{sql.NewOkResult(1)}}, }, { Query: "select i, ST_ASWKT(p) FROM tab2", Expected: []sql.Row{{1, "POINT(2 2)"}, {2, "POINT(1 6)"}}, }, { Query: "ALTER TABLE tab2 CHANGE COLUMN p p POINT NOT NULL SRID 4326", ExpectedErr: sql.ErrNotMatchingSRIDWithColName, }, { Query: "delete from tab2 where i = 1", Expected: []sql.Row{{sql.NewOkResult(1)}}, }, { Query: "ALTER TABLE tab2 CHANGE COLUMN p p POINT NOT NULL SRID 4326", Expected: []sql.Row{{sql.NewOkResult(0)}}, }, { Query: "show create table tab2", Expected: []sql.Row{{"tab2", "CREATE TABLE `tab2` (\n `i` int NOT NULL,\n `p` point NOT NULL SRID 4326,\n PRIMARY KEY (`i`)\n) ENGINE=InnoDB DEFAULT CHARSET=utf8mb4 COLLATE=utf8mb4_0900_bin"}}, }, }, }, { Name: "create table using SRID value for polygon type", SetUpScript: []string{ "CREATE TABLE tab3 (i int primary key, y polygon NOT NULL);", }, Assertions: []ScriptTestAssertion{ { Query: "show create table tab3", Expected: []sql.Row{{"tab3", "CREATE TABLE `tab3` (\n `i` int NOT NULL,\n `y` polygon NOT NULL,\n PRIMARY KEY (`i`)\n) ENGINE=InnoDB DEFAULT CHARSET=utf8mb4 COLLATE=utf8mb4_0900_bin"}}, }, { Query: "INSERT INTO tab3 VALUES (1, polygon(linestring(point(0,0),point(8,0),point(12,9),point(0,9),point(0,0))))", Expected: []sql.Row{{sql.NewOkResult(1)}}, }, { Query: "select i, ST_ASWKT(y) FROM tab3", Expected: []sql.Row{{1, "POLYGON((0 0,8 0,12 9,0 9,0 0))"}}, }, { Query: "ALTER TABLE tab3 MODIFY COLUMN y POLYGON NOT NULL SRID 0", Expected: []sql.Row{{sql.NewOkResult(0)}}, }, { Query: "ALTER TABLE tab3 MODIFY COLUMN y POLYGON NOT NULL SRID 4326", ExpectedErr: sql.ErrNotMatchingSRIDWithColName, }, { Query: "select i, ST_ASWKT(y) FROM tab3", Expected: []sql.Row{{1, "POLYGON((0 0,8 0,12 9,0 9,0 0))"}}, }, { Query: "ALTER TABLE tab3 MODIFY COLUMN y GEOMETRY NULL SRID 0", Expected: []sql.Row{{sql.NewOkResult(0)}}, }, { Query: "select i, ST_ASWKT(y) FROM tab3", Expected: []sql.Row{{1, "POLYGON((0 0,8 0,12 9,0 9,0 0))"}}, }, }, }, { Name: "invalid cases of SRID value", SetUpScript: []string{ "CREATE TABLE table1 (i int primary key, p point srid 4326);", "INSERT INTO table1 VALUES (1, ST_SRID(POINT(1, 5), 4326))", }, Assertions: []ScriptTestAssertion{ { Query: "CREATE TABLE table2 (i int primary key, p point srid 1);", ExpectedErr: sql.ErrUnsupportedFeature, }, { Query: "SELECT i, ST_ASWKT(p) FROM table1;", Expected: []sql.Row{{1, "POINT(5 1)"}}, }, { Query: "INSERT INTO table1 VALUES (2, POINT(2, 5))", ExpectedErr: sql.ErrNotMatchingSRIDWithColName, }, { Query: "SELECT i, ST_ASWKT(p) FROM table1;", Expected: []sql.Row{{1, "POINT(5 1)"}}, }, { Query: "ALTER TABLE table1 CHANGE COLUMN p p linestring srid 4326", ExpectedErr: sql.ErrSpatialTypeConversion, }, { Query: "ALTER TABLE table1 CHANGE COLUMN p p geometry srid 0", ExpectedErr: sql.ErrNotMatchingSRIDWithColName, }, { Query: "ALTER TABLE table1 CHANGE COLUMN p p geometry srid 4326", Expected: []sql.Row{{sql.NewOkResult(0)}}, }, { Query: "show create table table1", Expected: []sql.Row{{"table1", "CREATE TABLE `table1` (\n `i` int NOT NULL,\n `p` geometry SRID 4326,\n PRIMARY KEY (`i`)\n) ENGINE=InnoDB DEFAULT CHARSET=utf8mb4 COLLATE=utf8mb4_0900_bin"}}, }, { Query: "INSERT INTO table1 VALUES (2, ST_SRID(LINESTRING(POINT(0, 0),POINT(2, 2)),4326))", Expected: []sql.Row{{sql.NewOkResult(1)}}, }, { Query: "ALTER TABLE table1 CHANGE COLUMN p p point srid 4326", ExpectedErr: sql.ErrSpatialTypeConversion, }, }, }, }
var SpatialUpdateTests = []WriteQueryTest{ { WriteQuery: "UPDATE point_table SET p = point(123.456,789);", ExpectedWriteResult: []sql.Row{{newUpdateResult(1, 1)}}, SelectQuery: "SELECT * FROM point_table;", ExpectedSelect: []sql.Row{{int64(5), sql.Point{X: 123.456, Y: 789}}}, }, { WriteQuery: "UPDATE line_table SET l = linestring(point(1.2,3.4),point(5.6,7.8));", ExpectedWriteResult: []sql.Row{{newUpdateResult(2, 2)}}, SelectQuery: "SELECT * FROM line_table;", ExpectedSelect: []sql.Row{{int64(0), sql.LineString{Points: []sql.Point{{X: 1.2, Y: 3.4}, {X: 5.6, Y: 7.8}}}}, {int64(1), sql.LineString{Points: []sql.Point{{X: 1.2, Y: 3.4}, {X: 5.6, Y: 7.8}}}}}, }, { WriteQuery: "UPDATE polygon_table SET p = polygon(linestring(point(1,1),point(1,-1),point(-1,-1),point(-1,1),point(1,1)));", ExpectedWriteResult: []sql.Row{{newUpdateResult(1, 1)}}, SelectQuery: "SELECT * FROM polygon_table;", ExpectedSelect: []sql.Row{{int64(0), sql.Polygon{Lines: []sql.LineString{{Points: []sql.Point{{X: 1, Y: 1}, {X: 1, Y: -1}, {X: -1, Y: -1}, {X: -1, Y: 1}, {X: 1, Y: 1}}}}}}}, }, }
var TransactionTests = []TransactionTest{ { Name: "Changes from transactions are available before analyzing statements in other sessions (autocommit off)", Assertions: []ScriptTestAssertion{ { Query: "/* client a */ set @@autocommit = 0;", Expected: []sql.Row{{}}, }, { Query: "/* client b */ set @@autocommit = 0;", Expected: []sql.Row{{}}, }, { Query: "/* client a */ select @@autocommit;", Expected: []sql.Row{{0}}, }, { Query: "/* client b */ select @@autocommit;", Expected: []sql.Row{{0}}, }, { Query: "/* client a */ start transaction;", Expected: []sql.Row{}, }, { Query: "/* client a */ select * from t;", ExpectedErr: sql.ErrTableNotFound, }, { Query: "/* client a */ create table t(pk int primary key);", Expected: []sql.Row{{sql.OkResult{}}}, }, { Query: "/* client a */ select * from t2;", ExpectedErr: sql.ErrTableNotFound, }, { Query: "/* client a */ commit;", Expected: []sql.Row{}, }, { Query: "/* client b */ start transaction;", Expected: []sql.Row{}, }, { Query: "/* client b */ select count(*) from t;", Expected: []sql.Row{{0}}, }, }, }, { Name: "autocommit on", SetUpScript: []string{ "create table t (x int primary key, y int)", "insert into t values (1, 1)", }, Assertions: []ScriptTestAssertion{ { Query: "/* client a */ insert into t values (2, 2)", Expected: []sql.Row{{sql.NewOkResult(1)}}, }, { Query: "/* client b */ select * from t order by x", Expected: []sql.Row{{1, 1}, {2, 2}}, }, { Query: "/* client b */ insert into t values (3, 3)", Expected: []sql.Row{{sql.NewOkResult(1)}}, }, { Query: "/* client a */ select * from t order by x", Expected: []sql.Row{{1, 1}, {2, 2}, {3, 3}}, }, }, }, { Name: "autocommit off", SetUpScript: []string{ "create table t (x int primary key, y int)", "insert into t values (1, 1)", }, Assertions: []ScriptTestAssertion{ { Query: "/* client a */ set autocommit = off", Expected: []sql.Row{{}}, }, { Query: "/* client b */ set autocommit = off", Expected: []sql.Row{{}}, }, { Query: "/* client b */ select * from t order by x", Expected: []sql.Row{{1, 1}}, }, { Query: "/* client b */ insert into t values (2, 2)", Expected: []sql.Row{{sql.NewOkResult(1)}}, }, { Query: "/* client a */ select * from t order by x", Expected: []sql.Row{ {1, 1}, }, }, { Query: "/* client a */ insert into t values (3,3)", Expected: []sql.Row{{sql.NewOkResult(1)}}, }, { Query: "/* client b */ select * from t order by x", Expected: []sql.Row{{1, 1}, {2, 2}}, }, { Query: "/* client b */ commit", Expected: []sql.Row{}, }, { Query: "/* client a */ select * from t order by x", Expected: []sql.Row{{1, 1}, {3, 3}}, }, { Query: "/* client b */ select * from t order by x", Expected: []sql.Row{{1, 1}, {2, 2}}, }, { Query: "/* client a */ commit", Expected: []sql.Row{}, }, { Query: "/* client b */ select * from t order by x", Expected: []sql.Row{{1, 1}, {2, 2}}, }, { Query: "/* client b */ start transaction", Expected: []sql.Row{}, }, { Query: "/* client b */ select * from t order by x", Expected: []sql.Row{{1, 1}, {2, 2}, {3, 3}}, }, { Query: "/* client a */ select * from t order by x", Expected: []sql.Row{{1, 1}, {2, 2}, {3, 3}}, }, }, }, { Name: "toggle autocommit", SetUpScript: []string{ "create table t (x int primary key, y int)", "insert into t values (1, 1)", }, Assertions: []ScriptTestAssertion{ { Query: "/* client a */ set autocommit = off", Expected: []sql.Row{{}}, }, { Query: "/* client b */ set autocommit = off", Expected: []sql.Row{{}}, }, { Query: "/* client b */ insert into t values (2,2)", Expected: []sql.Row{{sql.NewOkResult(1)}}, }, { Query: "/* client a */ select * from t order by x", Expected: []sql.Row{{1, 1}}, }, { Query: "/* client b */ set autocommit = on", Expected: []sql.Row{{}}, }, { Query: "/* client a */ select * from t order by x", Expected: []sql.Row{{1, 1}}, }, { Query: "/* client a */ set autocommit = on", Expected: []sql.Row{{}}, }, { Query: "/* client a */ select * from t order by x", Expected: []sql.Row{{1, 1}, {2, 2}}, }, }, }, { Name: "autocommit on with explicit transactions", SetUpScript: []string{ "create table t (x int primary key, y int)", "insert into t values (1, 1)", }, Assertions: []ScriptTestAssertion{ { Query: "/* client a */ start transaction", Expected: []sql.Row{}, }, { Query: "/* client a */ select * from doesnotexist;", ExpectedErr: sql.ErrTableNotFound, }, { Query: "/* client a */ insert into t values (2, 2)", Expected: []sql.Row{{sql.NewOkResult(1)}}, }, { Query: "/* client b */ select * from t order by x", Expected: []sql.Row{{1, 1}}, }, { Query: "/* client a */ select * from doesnotexist;", ExpectedErr: sql.ErrTableNotFound, }, { Query: "/* client a */ commit", Expected: []sql.Row{}, }, { Query: "/* client b */ select * from t order by x", Expected: []sql.Row{{1, 1}, {2, 2}}, }, { Query: "/* client a */ insert into t values (3, 3)", Expected: []sql.Row{{sql.NewOkResult(1)}}, }, { Query: "/* client b */ select * from t order by x", Expected: []sql.Row{{1, 1}, {2, 2}, {3, 3}}, }, }, }, { Name: "rollback", SetUpScript: []string{ "create table t (x int primary key, y int)", "insert into t values (1, 1)", }, Assertions: []ScriptTestAssertion{ { Query: "/* client a */ set autocommit = off", Expected: []sql.Row{{}}, }, { Query: "/* client b */ set autocommit = off", Expected: []sql.Row{{}}, }, { Query: "/* client a */ start transaction", Expected: []sql.Row{}, }, { Query: "/* client b */ start transaction", Expected: []sql.Row{}, }, { Query: "/* client a */ insert into t values (2, 2)", Expected: []sql.Row{{sql.NewOkResult(1)}}, }, { Query: "/* client b */ insert into t values (3, 3)", Expected: []sql.Row{{sql.NewOkResult(1)}}, }, { Query: "/* client a */ select * from t order by x", Expected: []sql.Row{{1, 1}, {2, 2}}, }, { Query: "/* client b */ commit", Expected: []sql.Row{}, }, { Query: "/* client a */ select * from t order by x", Expected: []sql.Row{{1, 1}, {2, 2}}, }, { Query: "/* client a */ rollback", Expected: []sql.Row{}, }, { Query: "/* client a */ select * from t order by x", Expected: []sql.Row{{1, 1}, {3, 3}}, }, { Query: "/* client a */ insert into t values (2, 2)", Expected: []sql.Row{{sql.NewOkResult(1)}}, }, { Query: "/* client b */ select * from t order by x", Expected: []sql.Row{{1, 1}, {3, 3}}, }, { Query: "/* client a */ commit", Expected: []sql.Row{}, }, { Query: "/* client b */ select * from t order by x", Expected: []sql.Row{{1, 1}, {3, 3}}, }, { Query: "/* client b */ rollback", Expected: []sql.Row{}, }, { Query: "/* client b */ select * from t order by x", Expected: []sql.Row{{1, 1}, {2, 2}, {3, 3}}, }, }, }, { Name: "rollback to savepoint", SetUpScript: []string{ "create table t (x int primary key, y int)", "insert into t values (1, 1)", }, Assertions: []ScriptTestAssertion{ { Query: "/* client a */ set autocommit = off", Expected: []sql.Row{{}}, }, { Query: "/* client b */ set autocommit = off", Expected: []sql.Row{{}}, }, { Query: "/* client a */ start transaction", Expected: []sql.Row{}, }, { Query: "/* client b */ start transaction", Expected: []sql.Row{}, }, { Query: "/* client a */ insert into t values (2, 2)", Expected: []sql.Row{{sql.NewOkResult(1)}}, }, { Query: "/* client b */ insert into t values (3, 3)", Expected: []sql.Row{{sql.NewOkResult(1)}}, }, { Query: "/* client a */ savepoint spa1", Expected: []sql.Row{}, }, { Query: "/* client b */ savepoint spb1", Expected: []sql.Row{}, }, { Query: "/* client a */ insert into t values (4, 4)", Expected: []sql.Row{{sql.NewOkResult(1)}}, }, { Query: "/* client b */ insert into t values (5, 5)", Expected: []sql.Row{{sql.NewOkResult(1)}}, }, { Query: "/* client a */ savepoint spa2", Expected: []sql.Row{}, }, { Query: "/* client b */ savepoint spb2", Expected: []sql.Row{}, }, { Query: "/* client a */ insert into t values (6, 6)", Expected: []sql.Row{{sql.NewOkResult(1)}}, }, { Query: "/* client b */ insert into t values (7, 7)", Expected: []sql.Row{{sql.NewOkResult(1)}}, }, { Query: "/* client a */ select * from t order by x", Expected: []sql.Row{{1, 1}, {2, 2}, {4, 4}, {6, 6}}, }, { Query: "/* client b */ select * from t order by x", Expected: []sql.Row{{1, 1}, {3, 3}, {5, 5}, {7, 7}}, }, { Query: "/* client a */ rollback to SPA2", Expected: []sql.Row{}, }, { Query: "/* client b */ rollback to spB2", Expected: []sql.Row{}, }, { Query: "/* client a */ select * from t order by x", Expected: []sql.Row{{1, 1}, {2, 2}, {4, 4}}, }, { Query: "/* client b */ select * from t order by x", Expected: []sql.Row{{1, 1}, {3, 3}, {5, 5}}, }, { Query: "/* client a */ rollback to sPa2", Expected: []sql.Row{}, }, { Query: "/* client b */ rollback to Spb2", Expected: []sql.Row{}, }, { Query: "/* client a */ select * from t order by x", Expected: []sql.Row{{1, 1}, {2, 2}, {4, 4}}, }, { Query: "/* client b */ select * from t order by x", Expected: []sql.Row{{1, 1}, {3, 3}, {5, 5}}, }, { Query: "/* client a */ rollback to spA1", Expected: []sql.Row{}, }, { Query: "/* client b */ rollback to SPb1", Expected: []sql.Row{}, }, { Query: "/* client a */ select * from t order by x", Expected: []sql.Row{{1, 1}, {2, 2}}, }, { Query: "/* client b */ select * from t order by x", Expected: []sql.Row{{1, 1}, {3, 3}}, }, { Query: "/* client a */ rollback to spa2", ExpectedErr: sql.ErrSavepointDoesNotExist, }, { Query: "/* client b */ rollback to spb2", ExpectedErr: sql.ErrSavepointDoesNotExist, }, { Query: "/* client a */ rollback to Spa1", Expected: []sql.Row{}, }, { Query: "/* client b */ rollback to spB1", Expected: []sql.Row{}, }, { Query: "/* client a */ select * from t order by x", Expected: []sql.Row{{1, 1}, {2, 2}}, }, { Query: "/* client b */ select * from t order by x", Expected: []sql.Row{{1, 1}, {3, 3}}, }, { Query: "/* client a */ rollback", Expected: []sql.Row{}, }, { Query: "/* client b */ commit", Expected: []sql.Row{}, }, { Query: "/* client a */ select * from t order by x", Expected: []sql.Row{{1, 1}, {3, 3}}, }, { Query: "/* client b */ select * from t order by x", Expected: []sql.Row{{1, 1}, {3, 3}}, }, { Query: "/* client a */ rollback to spa1", ExpectedErr: sql.ErrSavepointDoesNotExist, }, { Query: "/* client b */ rollback to spb1", ExpectedErr: sql.ErrSavepointDoesNotExist, }, }, }, { Name: "release savepoint", SetUpScript: []string{ "create table t (x int primary key, y int)", "insert into t values (1, 1)", }, Assertions: []ScriptTestAssertion{ { Query: "/* client a */ set autocommit = off", Expected: []sql.Row{{}}, }, { Query: "/* client b */ set autocommit = off", Expected: []sql.Row{{}}, }, { Query: "/* client a */ start transaction", Expected: []sql.Row{}, }, { Query: "/* client b */ start transaction", Expected: []sql.Row{}, }, { Query: "/* client a */ insert into t values (2, 2)", Expected: []sql.Row{{sql.NewOkResult(1)}}, }, { Query: "/* client b */ insert into t values (3, 3)", Expected: []sql.Row{{sql.NewOkResult(1)}}, }, { Query: "/* client a */ savepoint spa1", Expected: []sql.Row{}, }, { Query: "/* client b */ savepoint spb1", Expected: []sql.Row{}, }, { Query: "/* client a */ release savepoint Spa1", Expected: []sql.Row{}, }, { Query: "/* client b */ release savepoint sPb1", Expected: []sql.Row{}, }, { Query: "/* client a */ rollback to spa1", ExpectedErr: sql.ErrSavepointDoesNotExist, }, { Query: "/* client b */ rollback to spb1", ExpectedErr: sql.ErrSavepointDoesNotExist, }, { Query: "/* client a */ select * from t order by x", Expected: []sql.Row{{1, 1}, {2, 2}}, }, { Query: "/* client b */ select * from t order by x", Expected: []sql.Row{{1, 1}, {3, 3}}, }, }, }, { Name: "overwrite savepoint", SetUpScript: []string{ "create table t (x int primary key, y int)", "insert into t values (1, 1)", }, Assertions: []ScriptTestAssertion{ { Query: "/* client a */ start transaction", Expected: []sql.Row{}, }, { Query: "/* client a */ insert into t values (2, 2)", Expected: []sql.Row{{sql.NewOkResult(1)}}, }, { Query: "/* client a */ savepoint spa1", Expected: []sql.Row{}, }, { Query: "/* client a */ insert into t values (3, 3)", Expected: []sql.Row{{sql.NewOkResult(1)}}, }, { Query: "/* client a */ savepoint spa2", Expected: []sql.Row{}, }, { Query: "/* client a */ insert into t values (4, 4)", Expected: []sql.Row{{sql.NewOkResult(1)}}, }, { Query: "/* client a */ savepoint SPA1", Expected: []sql.Row{}, }, { Query: "/* client a */ insert into t values (5, 5)", Expected: []sql.Row{{sql.NewOkResult(1)}}, }, { Query: "/* client a */ select * from t order by x", Expected: []sql.Row{{1, 1}, {2, 2}, {3, 3}, {4, 4}, {5, 5}}, }, { Query: "/* client a */ rollback to Spa1", Expected: []sql.Row{}, }, { Query: "/* client a */ select * from t order by x", Expected: []sql.Row{{1, 1}, {2, 2}, {3, 3}, {4, 4}}, }, { Query: "/* client a */ rollback to spa2", Expected: []sql.Row{}, }, { Query: "/* client a */ select * from t order by x", Expected: []sql.Row{{1, 1}, {2, 2}, {3, 3}}, }, { Query: "/* client a */ rollback to spa1", ExpectedErr: sql.ErrSavepointDoesNotExist, }, { Query: "/* client a */ release savepoint spa1", ExpectedErr: sql.ErrSavepointDoesNotExist, }, }, }, { Name: "Test AUTO INCREMENT with no autocommit", SetUpScript: []string{ "CREATE table t (x int PRIMARY KEY AUTO_INCREMENT, y int);", "CREATE table t2 (x int PRIMARY KEY AUTO_INCREMENT, y int);", "CREATE table t3 (x int PRIMARY KEY AUTO_INCREMENT, y int);", "insert into t (y) values (1);", "insert into t2 values (10, 10);", }, Assertions: []ScriptTestAssertion{ { Query: "/* client a */ set autocommit = off", Expected: []sql.Row{{}}, }, { Query: "/* client b */ set autocommit = off", Expected: []sql.Row{{}}, }, { Query: "/* client c */ set autocommit = off", Expected: []sql.Row{{}}, }, { Query: "/* client a */ insert into t (y) values (2)", Expected: []sql.Row{{sql.OkResult{RowsAffected: 1, InsertID: 2}}}, }, { Query: "/* client b */ select * from t order by x", Expected: []sql.Row{{1, 1}}, }, { Query: "/* client c*/ select * from t order by x", Expected: []sql.Row{{1, 1}}, }, { Query: "/* client b */ insert into t (y) values (3)", Expected: []sql.Row{{sql.OkResult{RowsAffected: 1, InsertID: 3}}}, }, { Query: "/* client a */ select * from t order by x", Expected: []sql.Row{ {1, 1}, {2, 2}, }, }, { Query: "/* client c */ select * from t order by x", Expected: []sql.Row{ {1, 1}, }, }, { Query: "/* client c */ insert into t2 (y) values (11)", Expected: []sql.Row{{sql.OkResult{RowsAffected: 1, InsertID: 11}}}, }, { Query: "/* client a */ select * from t2 order by x", Expected: []sql.Row{{10, 10}}, }, { Query: "/* client b */ select * from t2 order by x", Expected: []sql.Row{{10, 10}}, }, { Query: "/* client c */ select * from t2 order by x", Expected: []sql.Row{{10, 10}, {11, 11}}, }, { Query: "/* client a */ insert into t2 (y) values (12)", Expected: []sql.Row{{sql.OkResult{RowsAffected: 1, InsertID: 12}}}, }, { Query: "/* client a */ select * from t2 order by x", Expected: []sql.Row{{10, 10}, {12, 12}}, }, { Query: "/* client b */ select * from t2 order by x", Expected: []sql.Row{{10, 10}}, }, { Query: "/* client c */ select * from t2 order by x", Expected: []sql.Row{{10, 10}, {11, 11}}, }, { Query: "/* client a */ commit", Expected: []sql.Row{}, }, { Query: "/* client b */ select * from t order by x", Expected: []sql.Row{{1, 1}, {3, 3}}, }, { Query: "/* client b */ select * from t2 order by x", Expected: []sql.Row{{10, 10}}, }, { Query: "/* client c */ select * from t2 order by x", Expected: []sql.Row{{10, 10}, {11, 11}}, }, { Query: "/* client c */ select * from t order by x", Expected: []sql.Row{ {1, 1}, }, }, { Query: "/* client b */ commit", Expected: []sql.Row{}, }, { Query: "/* client a */ select * from t order by x", Expected: []sql.Row{{1, 1}, {2, 2}, {3, 3}}, }, { Query: "/* client a */ select * from t2 order by x", Expected: []sql.Row{{10, 10}, {12, 12}}, }, { Query: "/* client c */ select * from t2 order by x", Expected: []sql.Row{{10, 10}, {11, 11}}, }, { Query: "/* client c */ select * from t order by x", Expected: []sql.Row{{1, 1}}, }, { Query: "/* client c */ commit", Expected: []sql.Row{}, }, { Query: "/* client a */ select * from t order by x", Expected: []sql.Row{{1, 1}, {2, 2}, {3, 3}}, }, { Query: "/* client a */ select * from t2 order by x", Expected: []sql.Row{{10, 10}, {12, 12}}, }, { Query: "/* client b */ select * from t order by x", Expected: []sql.Row{{1, 1}, {2, 2}, {3, 3}}, }, { Query: "/* client b */ select * from t2 order by x", Expected: []sql.Row{{10, 10}, {11, 11}, {12, 12}}, }, { Query: "/* client a */ start transaction", Expected: []sql.Row{}, }, { Query: "/* client a */ select * from t order by x", Expected: []sql.Row{{1, 1}, {2, 2}, {3, 3}}, }, { Query: "/* client b */ select * from t order by x", Expected: []sql.Row{{1, 1}, {2, 2}, {3, 3}}, }, { Query: "/* client c */ select * from t order by x", Expected: []sql.Row{{1, 1}, {2, 2}, {3, 3}}, }, { Query: "/* client a */ select * from t2 order by x", Expected: []sql.Row{{10, 10}, {11, 11}, {12, 12}}, }, { Query: "/* client b */ select * from t2 order by x", Expected: []sql.Row{{10, 10}, {11, 11}, {12, 12}}, }, { Query: "/* client c */ select * from t2 order by x", Expected: []sql.Row{{10, 10}, {11, 11}, {12, 12}}, }, { Query: "/* client a */ insert into t values (10, 10)", Expected: []sql.Row{{sql.OkResult{RowsAffected: 1, InsertID: 10}}}, }, { Query: "/* client b */ insert into t (y) values (11)", Expected: []sql.Row{{sql.OkResult{RowsAffected: 1, InsertID: 11}}}, }, { Query: "/* client c */ insert into t values (50, 50)", Expected: []sql.Row{{sql.OkResult{RowsAffected: 1, InsertID: 50}}}, }, { Query: "/* client b */ insert into t (y) values (51)", Expected: []sql.Row{{sql.OkResult{RowsAffected: 1, InsertID: 51}}}, }, { Query: "/* client a */ select * from t order by x", Expected: []sql.Row{{1, 1}, {2, 2}, {3, 3}, {10, 10}}, }, { Query: "/* client b */ select * from t order by x", Expected: []sql.Row{{1, 1}, {2, 2}, {3, 3}, {11, 11}, {51, 51}}, }, { Query: "/* client c */ select * from t order by x", Expected: []sql.Row{{1, 1}, {2, 2}, {3, 3}, {50, 50}}, }, { Query: "/* client a */ commit", Expected: []sql.Row{}, }, { Query: "/* client b */ commit", Expected: []sql.Row{}, }, { Query: "/* client c */ commit", Expected: []sql.Row{}, }, { Query: "/* client a */ select * from t order by x", Expected: []sql.Row{{1, 1}, {2, 2}, {3, 3}, {10, 10}, {11, 11}, {50, 50}, {51, 51}}, }, { Query: "/* client b */ select * from t order by x", Expected: []sql.Row{{1, 1}, {2, 2}, {3, 3}, {10, 10}, {11, 11}, {50, 50}, {51, 51}}, }, { Query: "/* client c */ select * from t order by x", Expected: []sql.Row{{1, 1}, {2, 2}, {3, 3}, {10, 10}, {11, 11}, {50, 50}, {51, 51}}, }, { Query: "/* client a */ insert into t values (NULL, 52)", Expected: []sql.Row{{sql.OkResult{RowsAffected: 1, InsertID: 52}}}, }, { Query: "/* client a */ select * from t order by x", Expected: []sql.Row{{1, 1}, {2, 2}, {3, 3}, {10, 10}, {11, 11}, {50, 50}, {51, 51}, {52, 52}}, }, }, }, { Name: "AUTO_INCREMENT transactions off", SetUpScript: []string{ "CREATE table t2 (x int PRIMARY KEY AUTO_INCREMENT, y int);", "insert into t2 (y) values (1);", }, Assertions: []ScriptTestAssertion{ { Query: "/* client a */ insert into t2 (y) values (2)", Expected: []sql.Row{{sql.OkResult{RowsAffected: 1, InsertID: 2}}}, }, { Query: "/* client b */ select * from t2 order by x", Expected: []sql.Row{{1, 1}, {2, 2}}, }, { Query: "/* client b */ insert into t2 (y) values (3)", Expected: []sql.Row{{sql.OkResult{RowsAffected: 1, InsertID: 3}}}, }, { Query: "/* client a */ select * from t2 order by x", Expected: []sql.Row{{1, 1}, {2, 2}, {3, 3}}, }, }, }, { Name: "READ ONLY Transactions", SetUpScript: []string{ "create table t2 (pk int primary key, val int)", "insert into t2 values (0,0)", "commit", }, Assertions: []ScriptTestAssertion{ { Query: "/* client a */ set autocommit = off", Expected: []sql.Row{{}}, }, { Query: "/* client a */ create temporary table tmp(pk int primary key)", Expected: []sql.Row{{sql.NewOkResult(0)}}, }, { Query: "/* client a */ START TRANSACTION READ ONLY", Expected: []sql.Row{}, }, { Query: "/* client a */ INSERT INTO tmp VALUES (1)", Expected: []sql.Row{{sql.NewOkResult(1)}}, }, { Query: "/* client a */ insert into t2 values (1, 1)", ExpectedErr: sql.ErrReadOnlyTransaction, }, { Query: "/* client a */ insert into t2 values (2, 2)", ExpectedErr: sql.ErrReadOnlyTransaction, }, { Query: "/* client a */ delete from t2 where pk = 0", ExpectedErr: sql.ErrReadOnlyTransaction, }, { Query: "/* client a */ alter table t2 add val2 int", Expected: []sql.Row{{sql.NewOkResult(0)}}, }, { Query: "/* client a */ select * from t2", Expected: []sql.Row{{0, 0, nil}}, }, { Query: "/* client a */ create temporary table tmp2(pk int primary key)", ExpectedErr: sql.ErrReadOnlyTransaction, }, { Query: "/* client a */ COMMIT", Expected: []sql.Row{}, }, { Query: "/* client b */ START TRANSACTION READ ONLY", Expected: []sql.Row{}, }, { Query: "/* client b */ SELECT * FROM t2", Expected: []sql.Row{{0, 0, nil}}, }, }, }, }
var TriggerErrorTests = []ScriptTest{ { Name: "table doesn't exist", SetUpScript: []string{ "create table x (a int primary key, b int, c int)", }, Query: "create trigger not_found before insert on y for each row set new.a = new.a + 1", ExpectedErr: sql.ErrTableNotFound, }, { Name: "trigger errors on execution", SetUpScript: []string{ "create table x (a int primary key, b int)", "create table y (c int primary key not null)", "create trigger trigger_has_error before insert on x for each row insert into y values (null)", }, Query: "insert into x values (1,2)", ExpectedErr: sql.ErrInsertIntoNonNullableProvidedNull, }, { Name: "self update on insert", SetUpScript: []string{ "create table a (x int primary key)", "create trigger a1 before insert on a for each row insert into a values (new.x * 2)", }, Query: "insert into a values (1), (2), (3)", ExpectedErr: sql.ErrTriggerTableInUse, }, { Name: "self update on delete", SetUpScript: []string{ "create table a (x int primary key)", "create trigger a1 before delete on a for each row delete from a", }, Query: "delete from a", ExpectedErr: sql.ErrTriggerTableInUse, }, { Name: "self update on update", SetUpScript: []string{ "create table a (x int primary key)", "create trigger a1 before update on a for each row update a set x = 1", }, Query: "update a set x = 2", ExpectedErr: sql.ErrTriggerTableInUse, }, { Name: "circular dependency", SetUpScript: []string{ "create table a (x int primary key)", "create table b (y int primary key)", "create trigger a1 before insert on a for each row insert into b values (new.x * 2)", "create trigger b1 before insert on b for each row insert into a values (new.y * 7)", }, Query: "insert into a values (1), (2), (3)", ExpectedErr: sql.ErrTriggerTableInUse, }, { Name: "circular dependency, nested two deep", SetUpScript: []string{ "create table a (x int primary key)", "create table b (y int primary key)", "create table c (z int primary key)", "create trigger a1 before insert on a for each row insert into b values (new.x * 2)", "create trigger b1 before insert on b for each row insert into c values (new.y * 5)", "create trigger c1 before insert on c for each row insert into a values (new.z * 7)", }, Query: "insert into a values (1), (2), (3)", ExpectedErr: sql.ErrTriggerTableInUse, }, { Name: "reference to old on insert", SetUpScript: []string{ "create table x (a int primary key, b int, c int)", }, Query: "create trigger old_on_insert before insert on x for each row set new.c = old.a + 1", ExpectedErr: sql.ErrInvalidUseOfOldNew, }, { Name: "reference to new on delete", SetUpScript: []string{ "create table x (a int primary key, b int, c int)", }, Query: "create trigger new_on_delete before delete on x for each row set new.c = old.a + 1", ExpectedErr: sql.ErrInvalidUseOfOldNew, }, { Name: "set old row on update", SetUpScript: []string{ "create table x (a int primary key, b int, c int)", }, Query: "create trigger update_old before update on x for each row set old.c = new.a + 1", ExpectedErr: sql.ErrInvalidUpdateOfOldRow, }, { Name: "set old row on update, begin block", SetUpScript: []string{ "create table x (a int primary key, b int, c int)", }, Query: "create trigger update_old before update on x for each row BEGIN set old.c = new.a + 1; END", ExpectedErr: sql.ErrInvalidUpdateOfOldRow, }, { Name: "set new row after insert", SetUpScript: []string{ "create table x (a int primary key, b int, c int)", }, Query: "create trigger update_new after insert on x for each row set new.c = new.a + 1", ExpectedErr: sql.ErrInvalidUpdateInAfterTrigger, }, { Name: "set new row after update", SetUpScript: []string{ "create table x (a int primary key, b int, c int)", }, Query: "create trigger update_new after update on x for each row set new.c = new.a + 1", ExpectedErr: sql.ErrInvalidUpdateInAfterTrigger, }, { Name: "set new row after update, begin block", SetUpScript: []string{ "create table x (a int primary key, b int, c int)", }, Query: "create trigger update_new after update on x for each row BEGIN set new.c = new.a + 1; END", ExpectedErr: sql.ErrInvalidUpdateInAfterTrigger, }, { Name: "source column doesn't exist", SetUpScript: []string{ "create table x (a int primary key, b int, c int)", }, Query: "create trigger not_found before insert on x for each row set new.d = new.d + 1", ExpectedErr: sql.ErrUnknownColumn, }, { Name: "target column doesn't exist", SetUpScript: []string{ "create table x (a int primary key, b int, c int)", }, Query: "create trigger not_found before insert on x for each row set new.d = new.a + 1", ExpectedErr: sql.ErrUnknownColumn, }, }
var TriggerTests = []ScriptTest{ { Name: "trigger before inserts, use updated reference to other table", SetUpScript: []string{ "create table a (i int primary key, j int)", "create table b (x int primary key)", "create trigger trig before insert on a for each row begin set new.j = (select coalesce(max(x),1) from b); update b set x = x + 1; end;", "insert into b values (1)", "insert into a values (1,0), (2,0), (3,0)", }, Assertions: []ScriptTestAssertion{ { Query: "select * from a order by i", Expected: []sql.Row{ {1, 1}, {2, 2}, {3, 3}, }, }, { Query: "select x from b", Expected: []sql.Row{ {4}, }, }, { Query: "insert into a values (4,0), (5,0)", Expected: []sql.Row{ {sql.OkResult{RowsAffected: 2}}, }, }, }, }, { Name: "trigger before inserts, use count updated reference to other table", SetUpScript: []string{ "create table a (i int, j int)", "create table b (x int)", "create trigger trig before insert on a for each row begin set new.j = (select count(x) from b); insert into b values (new.i + new.j); end;", "insert into a values (0,0), (0,0), (0,0)", }, Assertions: []ScriptTestAssertion{ { Query: "select * from a order by j", Expected: []sql.Row{ {0, 0}, {0, 1}, {0, 2}, }, }, { Query: "select x from b", Expected: []sql.Row{ {0}, {1}, {2}, }, }, { Query: "insert into a values (0,0), (0,0)", Expected: []sql.Row{ {sql.OkResult{RowsAffected: 2}}, }, }, }, }, { Name: "trigger after insert, insert into other table", SetUpScript: []string{ "create table a (x int primary key)", "create table b (y int primary key)", "create trigger insert_into_b after insert on a for each row insert into b values (new.x + 1)", "insert into a values (1), (3), (5)", }, Assertions: []ScriptTestAssertion{ { Query: "select x from a order by 1", Expected: []sql.Row{ {1}, {3}, {5}, }, }, { Query: "select y from b order by 1", Expected: []sql.Row{ {2}, {4}, {6}, }, }, { Query: "insert into a values (7), (9)", Expected: []sql.Row{ {sql.OkResult{RowsAffected: 2}}, }, }, }, }, { Name: "trigger after insert, delete from other table", SetUpScript: []string{ "create table a (x int primary key)", "create table b (y int primary key)", "insert into b values (0), (2), (4), (6), (8)", "create trigger insert_into_b after insert on a for each row delete from b where y = (new.x + 1)", "insert into a values (1), (3), (5)", }, Assertions: []ScriptTestAssertion{ { Query: "select x from a order by 1", Expected: []sql.Row{ {1}, {3}, {5}, }, }, { Query: "select y from b order by 1", Expected: []sql.Row{ {0}, {8}, }, }, { Query: "insert into a values (7), (9)", Expected: []sql.Row{ {sql.OkResult{RowsAffected: 2}}, }, }, }, }, { Name: "trigger after insert, update other table", SetUpScript: []string{ "create table a (x int primary key)", "create table b (y int primary key)", "insert into b values (0), (2), (4), (6), (8)", "create trigger insert_into_b after insert on a for each row update b set y = new.x where y = new.x + 1", "insert into a values (1), (3), (5)", }, Assertions: []ScriptTestAssertion{ { Query: "select x from a order by 1", Expected: []sql.Row{ {1}, {3}, {5}, }, }, { Query: "select y from b order by 1", Expected: []sql.Row{ {0}, {1}, {3}, {5}, {8}, }, }, }, }, { Name: "trigger before insert, insert into other table", SetUpScript: []string{ "create table a (x int primary key)", "create table b (y int primary key)", "create trigger insert_into_b before insert on a for each row insert into b values (new.x + 1)", "insert into a values (1), (3), (5)", }, Assertions: []ScriptTestAssertion{ { Query: "select x from a order by 1", Expected: []sql.Row{ {1}, {3}, {5}, }, }, { Query: "select y from b order by 1", Expected: []sql.Row{ {2}, {4}, {6}, }, }, { Query: "insert into a values (7), (9)", Expected: []sql.Row{ {sql.OkResult{RowsAffected: 2}}, }, }, }, }, { Name: "trigger before insert, insert into other table with different schema", SetUpScript: []string{ "create table a (x int primary key, y int)", "create table b (z int primary key)", "create trigger insert_into_b before insert on a for each row insert into b values (new.x + 1)", "insert into a values (1,2), (3,4), (5,6)", }, Assertions: []ScriptTestAssertion{ { Query: "select x from a order by 1", Expected: []sql.Row{ {1}, {3}, {5}, }, }, { Query: "select z from b order by 1", Expected: []sql.Row{ {2}, {4}, {6}, }, }, { Query: "insert into a values (7,8), (9,10)", Expected: []sql.Row{ {sql.OkResult{RowsAffected: 2}}, }, }, }, }, { Name: "trigger before insert, delete from other table", SetUpScript: []string{ "create table a (x int primary key)", "create table b (y int primary key)", "insert into b values (0), (2), (4), (6), (8)", "create trigger insert_into_b before insert on a for each row delete from b where y = (new.x + 1)", "insert into a values (1), (3), (5)", }, Assertions: []ScriptTestAssertion{ { Query: "select x from a order by 1", Expected: []sql.Row{ {1}, {3}, {5}, }, }, { Query: "select y from b order by 1", Expected: []sql.Row{ {0}, {8}, }, }, { Query: "insert into a values (7), (9)", Expected: []sql.Row{ {sql.OkResult{RowsAffected: 2}}, }, }, }, }, { Name: "trigger before insert, update other table", SetUpScript: []string{ "create table a (x int primary key)", "create table b (y int primary key)", "insert into b values (0), (2), (4), (6), (8)", "create trigger insert_into_b before insert on a for each row update b set y = new.x where y = new.x + 1", "insert into a values (1), (3), (5)", }, Assertions: []ScriptTestAssertion{ { Query: "select x from a order by 1", Expected: []sql.Row{ {1}, {3}, {5}, }, }, { Query: "select y from b order by 1", Expected: []sql.Row{ {0}, {1}, {3}, {5}, {8}, }, }, }, }, { Name: "trigger before insert, updates references to 2 tables", SetUpScript: []string{ "create table a (i int, j int, k int)", "create table b (x int)", "create table c (y int)", "insert into b values (0)", "insert into c values (0)", "create trigger trig before insert on a for each row begin set new.j = (select x from b); set new.k = (select y from c); update b set x = x + 1; update c set y = y + 2; end;", "insert into a values (0, 0, 0), (1, 0, 0), (2, 0, 0), (3, 0, 0), (4, 0, 0)", }, Assertions: []ScriptTestAssertion{ { Query: "select * from a order by 1", Expected: []sql.Row{ {0, 0, 0}, {1, 1, 2}, {2, 2, 4}, {3, 3, 6}, {4, 4, 8}, }, }, { Query: "select x from b order by 1", Expected: []sql.Row{ {5}, }, }, { Query: "select y from c order by 1", Expected: []sql.Row{ {10}, }, }, }, }, { Name: "trigger before insert, alter inserted value", SetUpScript: []string{ "create table a (x int primary key)", "create trigger insert_into_a before insert on a for each row set new.x = new.x + 1", "insert into a values (1)", }, Query: "select x from a order by 1", Expected: []sql.Row{ {2}, }, }, { Name: "trigger before insert, alter inserted value, multiple columns", SetUpScript: []string{ "create table x (a int primary key, b int, c int)", "create trigger insert_into_x before insert on x for each row set new.a = new.a + 1, new.b = new.c, new.c = 0", "insert into x values (1, 10, 100)", }, Query: "select * from x order by 1", Expected: []sql.Row{ {2, 100, 0}, }, }, { Name: "trigger before insert, alter inserted value, multiple columns, system var", SetUpScript: []string{ "create table x (a int primary key, b int, c int)", "set @@auto_increment_increment = 1", "create trigger insert_into_x before insert on x for each row " + "set new.a = new.a + 1, new.b = new.c, new.c = 0, @@auto_increment_increment = @@auto_increment_increment + 1", "insert into x values (1, 10, 100), (2, 20, 200)", }, Query: "select *, @@auto_increment_increment from x order by 1", Expected: []sql.Row{ {2, 100, 0, 3}, {3, 200, 0, 3}, }, }, { Name: "trigger before insert, alter inserted value, out of order insertion", SetUpScript: []string{ "create table a (x int primary key, y int)", "create trigger a1 before insert on a for each row set new.x = new.x * 2, new.y = new.y * 3", }, Assertions: []ScriptTestAssertion{ { Query: "insert into a (y, x) values (5,7), (9,11)", Expected: []sql.Row{ {sql.OkResult{RowsAffected: 2}}, }, }, { Query: "select x, y from a order by 1", Expected: []sql.Row{ {14, 15}, {22, 27}, }, }, }, }, { Name: "trigger before insert, alter inserted value, incomplete insertion", SetUpScript: []string{ "create table a (x int primary key, y int, z int default 5)", "create trigger a1 before insert on a for each row set new.x = new.x * 2, new.y = new.y * 3, new.z = new.z * 5", }, Assertions: []ScriptTestAssertion{ { Query: "insert into a (y, x) values (5,7), (9,11)", Expected: []sql.Row{ {sql.OkResult{RowsAffected: 2}}, }, }, { Query: "select x, y, z from a order by 1", Expected: []sql.Row{ {14, 15, 25}, {22, 27, 25}, }, }, }, }, { Name: "trigger before insert, begin block with multiple set statements", SetUpScript: []string{ "CREATE TABLE test(pk BIGINT PRIMARY KEY, v1 BIGINT);", "INSERT INTO test VALUES (0,2),(1,3)", `CREATE TRIGGER tt BEFORE INSERT ON test FOR EACH ROW BEGIN SET NEW.v1 = NEW.v1 * 11; SET NEW.v1 = NEW.v1 * -10; END;`, "INSERT INTO test VALUES (2,4), (6,8);", }, Assertions: []ScriptTestAssertion{ { Query: "SELECT * FROM test ORDER BY 1", Expected: []sql.Row{ {0, 2}, {1, 3}, {2, -440}, {6, -880}, }, }, }, }, { Name: "trigger before insert, begin block with multiple set statements and inserts", SetUpScript: []string{ "CREATE TABLE test(pk BIGINT PRIMARY KEY, v1 BIGINT);", "CREATE TABLE test2(pk BIGINT PRIMARY KEY, v1 BIGINT);", "CREATE TABLE test3(pk BIGINT PRIMARY KEY, v1 BIGINT);", "INSERT INTO test VALUES (0,2),(1,3)", `CREATE TRIGGER tt BEFORE INSERT ON test FOR EACH ROW BEGIN SET NEW.v1 = NEW.v1 * 11; insert into test2 values (new.pk * 3, new.v1); SET NEW.v1 = NEW.v1 * -10; insert into test3 values (new.pk * 5, new.v1); set @var = 0; END;`, "INSERT INTO test VALUES (2,4), (6,8);", }, Assertions: []ScriptTestAssertion{ { Query: "SELECT * FROM test ORDER BY 1", Expected: []sql.Row{ {0, 2}, {1, 3}, {2, -440}, {6, -880}, }, }, { Query: "SELECT * FROM test2 ORDER BY 1", Expected: []sql.Row{ {6, 44}, {18, 88}, }, }, { Query: "SELECT * FROM test3 ORDER BY 1", Expected: []sql.Row{ {10, -440}, {30, -880}, }, }, }, }, { Name: "Create a trigger on a new database and verify that the trigger works when selected on another database", SetUpScript: []string{ "create table foo.a (x int primary key)", "create table foo.b (y int primary key)", "use foo", "create trigger insert_into_b after insert on foo.a for each row insert into foo.b values (new.x + 1)", "use mydb", "insert into foo.a values (1), (3), (5)", }, Assertions: []ScriptTestAssertion{ { Query: "select x from foo.a order by 1", Expected: []sql.Row{ {1}, {3}, {5}, }, }, { Query: "select y from foo.b order by 1", Expected: []sql.Row{ {2}, {4}, {6}, }, }, { Query: "insert into foo.a values (7), (9)", Expected: []sql.Row{ {sql.OkResult{RowsAffected: 2}}, }, }, }, }, { Name: "trigger after update, insert into other table", SetUpScript: []string{ "create table a (x int primary key)", "create table b (y int primary key)", "insert into a values (1), (3), (5)", "create trigger insert_into_b after update on a for each row insert into b values (old.x + new.x + 1)", "update a set x = x + 1 where x in (1, 3)", }, Assertions: []ScriptTestAssertion{ { Query: "select x from a order by 1", Expected: []sql.Row{ {2}, {4}, {5}, }, }, { Query: "select y from b order by 1", Expected: []sql.Row{ {4}, {8}, }, }, { Query: "update a set x = x + 1 where x = 5", Expected: []sql.Row{ {sql.OkResult{ RowsAffected: 1, Info: plan.UpdateInfo{ Matched: 1, Updated: 1, }, }}, }, }, }, }, { Name: "trigger after update, delete from other table", SetUpScript: []string{ "create table a (x int primary key)", "create table b (y int primary key)", "insert into a values (0), (2), (4), (6), (8)", "insert into b values (1), (3), (5), (7), (9)", "create trigger delete_from_b after update on a for each row delete from b where y = old.x + new.x", "update a set x = x + 1 where x in (2,4)", }, Assertions: []ScriptTestAssertion{ { Query: "select x from a order by 1", Expected: []sql.Row{ {0}, {3}, {5}, {6}, {8}, }, }, { Query: "select y from b order by 1", Expected: []sql.Row{ {1}, {3}, {7}, }, }, }, }, { Name: "trigger after update, update other table", SetUpScript: []string{ "create table a (x int primary key)", "create table b (y int primary key)", "insert into a values (0), (2), (4), (6), (8)", "insert into b values (0), (2), (4), (8)", "create trigger update_b after update on a for each row update b set y = old.x + new.x + 1 where y = old.x", "update a set x = x + 1 where x in (2, 4)", }, Assertions: []ScriptTestAssertion{ { Query: "select x from a order by 1", Expected: []sql.Row{ {0}, {3}, {5}, {6}, {8}, }, }, { Query: "select y from b order by 1", Expected: []sql.Row{ {0}, {6}, {8}, {10}, }, }, }, }, { Name: "trigger before update, insert into other table", SetUpScript: []string{ "create table a (x int primary key)", "create table b (y int primary key)", "insert into a values (1), (3), (5)", "create trigger insert_into_b before update on a for each row insert into b values (old.x + new.x + 1)", "update a set x = x + 1 where x in (1, 3)", }, Assertions: []ScriptTestAssertion{ { Query: "select x from a order by 1", Expected: []sql.Row{ {2}, {4}, {5}, }, }, { Query: "select y from b order by 1", Expected: []sql.Row{ {4}, {8}, }, }, { Query: "update a set x = x + 1 where x = 5", Expected: []sql.Row{ {sql.OkResult{ RowsAffected: 1, Info: plan.UpdateInfo{ Matched: 1, Updated: 1, }, }}, }, }, }, }, { Name: "trigger before update, delete from other table", SetUpScript: []string{ "create table a (x int primary key)", "create table b (y int primary key)", "insert into a values (0), (2), (4), (6), (8)", "insert into b values (1), (3), (5), (7), (9)", "create trigger delete_from_b before update on a for each row delete from b where y = old.x + new.x", "update a set x = x + 1 where x in (2,4)", }, Assertions: []ScriptTestAssertion{ { Query: "select x from a order by 1", Expected: []sql.Row{ {0}, {3}, {5}, {6}, {8}, }, }, { Query: "select y from b order by 1", Expected: []sql.Row{ {1}, {3}, {7}, }, }, }, }, { Name: "trigger before update, update other table", SetUpScript: []string{ "create table a (x int primary key)", "create table b (y int primary key)", "insert into a values (0), (2), (4), (6), (8)", "insert into b values (0), (2), (4), (8)", "create trigger update_b before update on a for each row update b set y = old.x + new.x + 1 where y = old.x", "update a set x = x + 1 where x in (2, 4)", }, Assertions: []ScriptTestAssertion{ { Query: "select x from a order by 1", Expected: []sql.Row{ {0}, {3}, {5}, {6}, {8}, }, }, { Query: "select y from b order by 1", Expected: []sql.Row{ {0}, {6}, {8}, {10}, }, }, }, }, { Name: "trigger before update, set new value", SetUpScript: []string{ "create table a (x int primary key)", "insert into a values (1), (10)", "create trigger update_a before update on a for each row set new.x = new.x + old.x", "update a set x = x + 1", }, Query: "select x from a order by 1", Expected: []sql.Row{ {3}, {21}, }, }, { Name: "trigger before update, set new value to old value", SetUpScript: []string{ "create table a (x int primary key)", "insert into a values (1), (10)", "create trigger no_step_on_snek before update on a for each row set new.x = old.x", "update a set x = x + 1", }, Query: "select x from a order by 1", Expected: []sql.Row{ {1}, {10}, }, }, { Name: "trigger before update, set new values, multiple cols", SetUpScript: []string{ "create table a (x int primary key, y int)", "insert into a values (1,3), (10,20)", "create trigger update_a before update on a for each row set new.x = new.x + old.y, new.y = new.y + old.x", "update a set x = x + 1, y = y + 1", }, Query: "select x, y from a order by 1", Expected: []sql.Row{ {5, 5}, {31, 31}, }, }, { Name: "trigger before update, set new values, multiple cols (2)", SetUpScript: []string{ "create table a (x int primary key, y int)", "insert into a values (1,3), (10,20)", "create trigger update_a before update on a for each row set new.x = new.x + new.y, new.y = new.y + old.y", "update a set x = x + 1, y = y + 1", }, Query: "select x, y from a order by 1", Expected: []sql.Row{ {6, 7}, {32, 41}, }, }, { Name: "trigger before update, with indexed update", SetUpScript: []string{ "create table a (x int primary key, y int, unique key (y))", "create table b (z int primary key)", "insert into a values (1,3), (10,20)", "create trigger insert_b before update on a for each row insert into b values (old.x * 10)", "update a set x = x + 1 where y = 20", }, Assertions: []ScriptTestAssertion{ { Query: "select x, y from a order by 1", Expected: []sql.Row{ {1, 3}, {11, 20}, }, }, { Query: "select z from b", Expected: []sql.Row{ {100}, }, }, }, }, { Name: "trigger before update, begin block with multiple set statements", SetUpScript: []string{ "CREATE TABLE test(pk BIGINT PRIMARY KEY, v1 BIGINT);", "INSERT INTO test VALUES (0,2),(1,3)", "CREATE TRIGGER tt BEFORE UPDATE ON test FOR EACH ROW BEGIN SET NEW.v1 = (OLD.v1 * 2) + NEW.v1; SET NEW.v1 = NEW.v1 * -10; END;", "UPDATE test SET v1 = v1 + 1;", }, Assertions: []ScriptTestAssertion{ { Query: "SELECT * FROM test ORDER BY 1", Expected: []sql.Row{ {0, -70}, {1, -100}, }, }, }, }, { Name: "trigger after delete, insert into other table", SetUpScript: []string{ "create table a (x int primary key)", "create table b (y int primary key)", "insert into a values (1), (3), (5)", "create trigger insert_into_b after delete on a for each row insert into b values (old.x + 1)", "delete from a where x in (1, 3)", }, Assertions: []ScriptTestAssertion{ { Query: "select x from a order by 1", Expected: []sql.Row{ {5}, }, }, { Query: "select y from b order by 1", Expected: []sql.Row{ {2}, {4}, }, }, { Query: "delete from a where x = 5", Expected: []sql.Row{ {sql.OkResult{RowsAffected: 1}}, }, }, }, }, { Name: "trigger after delete, delete from other table", SetUpScript: []string{ "create table a (x int primary key)", "create table b (y int primary key)", "insert into a values (0), (2), (4), (6), (8)", "insert into b values (0), (2), (4), (6), (8)", "create trigger delete_from_b after delete on a for each row delete from b where y = old.x", "delete from a where x in (2,4,6)", }, Assertions: []ScriptTestAssertion{ { Query: "select x from a order by 1", Expected: []sql.Row{ {0}, {8}, }, }, { Query: "select y from b order by 1", Expected: []sql.Row{ {0}, {8}, }, }, }, }, { Name: "trigger after delete, update other table", SetUpScript: []string{ "create table a (x int primary key)", "create table b (y int primary key)", "insert into a values (0), (2), (4), (6), (8)", "insert into b values (0), (2), (4), (6), (8)", "create trigger update_b after delete on a for each row update b set y = old.x + 1 where y = old.x", "delete from a where x in (2,4,6)", }, Assertions: []ScriptTestAssertion{ { Query: "select x from a order by 1", Expected: []sql.Row{ {0}, {8}, }, }, { Query: "select y from b order by 1", Expected: []sql.Row{ {0}, {3}, {5}, {7}, {8}, }, }, }, }, { Name: "trigger before delete, insert into other table", SetUpScript: []string{ "create table a (x int primary key)", "create table b (y int primary key)", "insert into a values (0), (2), (4), (6), (8)", "create trigger insert_into_b before delete on a for each row insert into b values (old.x + 1)", "delete from a where x in (2, 4, 6)", }, Assertions: []ScriptTestAssertion{ { Query: "select x from a order by 1", Expected: []sql.Row{ {0}, {8}, }, }, { Query: "select y from b order by 1", Expected: []sql.Row{ {3}, {5}, {7}, }, }, { Query: "delete from a where x = 0", Expected: []sql.Row{ {sql.OkResult{RowsAffected: 1}}, }, }, }, }, { Name: "trigger before delete, delete from other table", SetUpScript: []string{ "create table a (x int primary key)", "create table b (y int primary key)", "insert into a values (0), (2), (4), (6), (8)", "insert into b values (1), (3), (5), (7), (9)", "create trigger delete_from_b before delete on a for each row delete from b where y = (old.x + 1)", "delete from a where x in (2, 4, 6)", }, Assertions: []ScriptTestAssertion{ { Query: "select x from a order by 1", Expected: []sql.Row{ {0}, {8}, }, }, { Query: "select y from b order by 1", Expected: []sql.Row{ {1}, {9}, }, }, }, }, { Name: "trigger before delete, update other table", SetUpScript: []string{ "create table a (x int primary key)", "create table b (y int primary key)", "insert into a values (0), (2), (4), (6), (8)", "insert into b values (1), (3), (5), (7), (9)", "create trigger update_b before delete on a for each row update b set y = old.x where y = old.x + 1", "delete from a where x in (2, 4, 6)", }, Assertions: []ScriptTestAssertion{ { Query: "select x from a order by 1", Expected: []sql.Row{ {0}, {8}, }, }, { Query: "select y from b order by 1", Expected: []sql.Row{ {1}, {2}, {4}, {6}, {9}, }, }, }, }, { Name: "trigger before delete, delete with index", SetUpScript: []string{ "create table a (x int primary key, z int, unique key (z))", "create table b (y int primary key)", "insert into a values (0,1), (2,3), (4,5)", "create trigger insert_b before delete on a for each row insert into b values (old.x * 2)", "delete from a where z > 2", }, Assertions: []ScriptTestAssertion{ { Query: "select x from a order by 1", Expected: []sql.Row{ {0}, }, }, { Query: "select y from b order by 1", Expected: []sql.Row{ {4}, {8}, }, }, }, }, { Name: "trigger before delete, update other table", SetUpScript: []string{ "create table a (i int primary key, j int)", "insert into a values (0,1), (2,3), (4,5)", "create table b (x int)", "insert into b values (0)", "create trigger trig before delete on a for each row begin update b set x = x + old.j; end;", "delete from a where true", }, Assertions: []ScriptTestAssertion{ { Query: "select * from a order by 1", Expected: []sql.Row{}, }, { Query: "select x from b order by 1", Expected: []sql.Row{ {9}, }, }, }, }, { Name: "triggers before and after insert", SetUpScript: []string{ "create table a (x int primary key)", "create table b (y int primary key)", "create trigger a1 before insert on a for each row insert into b values (NEW.x * 7)", "create trigger a2 after insert on a for each row insert into b values (New.x * 11)", }, Assertions: []ScriptTestAssertion{ { Query: "insert into a values (2), (3), (5)", Expected: []sql.Row{ {sql.NewOkResult(3)}, }, }, { Query: "select x from a order by 1", Expected: []sql.Row{ {2}, {3}, {5}, }, }, { Query: "select y from b order by 1", Expected: []sql.Row{ {14}, {21}, {22}, {33}, {35}, {55}, }, }, }, }, { Name: "multiple triggers before insert", SetUpScript: []string{ "create table a (x int primary key)", "create trigger a1 before insert on a for each row set new.x = New.x + 1", "create trigger a2 before insert on a for each row set new.x = New.x * 2", "create trigger a3 before insert on a for each row set new.x = New.x - 5", }, Assertions: []ScriptTestAssertion{ { Query: "insert into a values (1), (3)", Expected: []sql.Row{ {sql.NewOkResult(2)}, }, }, { Query: "select x from a order by 1", Expected: []sql.Row{ {-1}, {3}, }, }, }, }, { Name: "multiple triggers before insert, with precedes / follows", SetUpScript: []string{ "create table a (x int primary key)", "create trigger a1 before insert on a for each row set new.x = New.x + 1", "create trigger a2 before insert on a for each row precedes a1 set new.x = New.x * 2", "create trigger a3 before insert on a for each row precedes a2 set new.x = New.x - 5", "create trigger a4 before insert on a for each row follows a2 set new.x = New.x * 3", }, Assertions: []ScriptTestAssertion{ { Query: "insert into a values (1), (3)", Expected: []sql.Row{ {sql.NewOkResult(2)}, }, }, { Query: "select x from a order by 1", Expected: []sql.Row{ {-23}, {-11}, }, }, }, }, { Name: "triggers before and after update", SetUpScript: []string{ "create table a (x int primary key)", "create table b (y int primary key)", "create trigger a1 before update on a for each row insert into b values (old.x * 7)", "create trigger a2 after update on a for each row insert into b values (old.x * 11)", "insert into a values (2), (3), (5)", }, Assertions: []ScriptTestAssertion{ { Query: "update a set x = x * 2", Expected: []sql.Row{ {sql.OkResult{ RowsAffected: 3, Info: plan.UpdateInfo{ Matched: 3, Updated: 3, }, }}, }, }, { Query: "select x from a order by 1", Expected: []sql.Row{ {4}, {6}, {10}, }, }, { Query: "select y from b order by 1", Expected: []sql.Row{ {14}, {21}, {22}, {33}, {35}, {55}, }, }, }, }, { Name: "multiple triggers before and after update", SetUpScript: []string{ "create table a (x int primary key)", "create table b (y int primary key)", "create trigger a1 before update on a for each row insert into b values (old.x * 7)", "create trigger a2 after update on a for each row insert into b values (old.x * 11)", "create trigger a3 before update on a for each row insert into b values (old.x * 13)", "create trigger a4 after update on a for each row insert into b values (old.x * 17)", "insert into a values (2), (3), (5)", }, Assertions: []ScriptTestAssertion{ { Query: "update a set x = x * 2", Expected: []sql.Row{ {sql.OkResult{ RowsAffected: 3, Info: plan.UpdateInfo{ Matched: 3, Updated: 3, }, }}, }, }, { Query: "select x from a order by 1", Expected: []sql.Row{ {4}, {6}, {10}, }, }, { Query: "select y from b order by 1", Expected: []sql.Row{ {14}, {21}, {22}, {26}, {33}, {34}, {35}, {39}, {51}, {55}, {65}, {85}, }, }, }, }, { Name: "triggers before and after delete", SetUpScript: []string{ "create table a (x int primary key)", "create table b (y int primary key)", "create trigger a1 before delete on a for each row insert into b values (old.x * 7)", "create trigger a2 after delete on a for each row insert into b values (old.x * 11)", "insert into a values (2), (3), (5)", }, Assertions: []ScriptTestAssertion{ { Query: "delete from a", Expected: []sql.Row{ {sql.NewOkResult(3)}, }, }, { Query: "select x from a order by 1", Expected: []sql.Row{}, }, { Query: "select y from b order by 1", Expected: []sql.Row{ {14}, {21}, {22}, {33}, {35}, {55}, }, }, }, }, { Name: "multiple triggers before and after delete", SetUpScript: []string{ "create table a (x int primary key)", "create table b (y int primary key)", "create trigger a1 before delete on a for each row insert into b values (old.x * 7)", "create trigger a2 after delete on a for each row insert into b values (old.x * 11)", "create trigger a3 before delete on a for each row insert into b values (old.x * 13)", "create trigger a4 after delete on a for each row insert into b values (old.x * 17)", "insert into a values (2), (3), (5)", }, Assertions: []ScriptTestAssertion{ { Query: "delete from a", Expected: []sql.Row{ {sql.NewOkResult(3)}, }, }, { Query: "select x from a order by 1", Expected: []sql.Row{}, }, { Query: "select y from b order by 1", Expected: []sql.Row{ {14}, {21}, {22}, {26}, {33}, {34}, {35}, {39}, {51}, {55}, {65}, {85}, }, }, }, }, { Name: "multiple triggers before and after insert, with precedes / follows", SetUpScript: []string{ "create table a (x int primary key)", "create table b (y int primary key)", "insert into b values (1), (3)", "create trigger a1 before insert on a for each row set new.x = New.x + 1", "create trigger a2 before insert on a for each row precedes a1 set new.x = New.x * 2", "create trigger a3 before insert on a for each row precedes a2 set new.x = New.x - 5", "create trigger a4 before insert on a for each row follows a2 set new.x = New.x * 3", "create trigger a5 after insert on a for each row update b set y = y + 1 order by y asc", "create trigger a6 after insert on a for each row precedes a5 update b set y = y * 2 order by y asc", "create trigger a7 after insert on a for each row precedes a6 update b set y = y - 5 order by y asc", "create trigger a8 after insert on a for each row follows a6 update b set y = y * 3 order by y asc", }, Assertions: []ScriptTestAssertion{ { Query: "insert into a values (1), (3)", Expected: []sql.Row{ {sql.NewOkResult(2)}, }, }, { Query: "select x from a order by 1", Expected: []sql.Row{ {-23}, {-11}, }, }, { Query: "select y from b order by 1", Expected: []sql.Row{ {-167}, {-95}, }, }, }, }, { Name: "triggered update query which could project", SetUpScript: []string{ "create table trigger_on_update (id int primary key, first varchar(25), last varchar(25))", "create table is_dirty (id int primary key, is_dirty bool)", "insert into is_dirty values (1, false)", "insert into trigger_on_update values (1, 'george', 'smith')", `create trigger trigger_on_update_on_update before update on trigger_on_update for each row begin update is_dirty set is_dirty = true; end;`, }, Assertions: []ScriptTestAssertion{ { Query: "select id, is_dirty from is_dirty", Expected: []sql.Row{ {1, 0}, }, }, { Query: "update trigger_on_update set id = 1, first = 'george', last = 'smith' where id = 1", Expected: []sql.Row{ { sql.OkResult{ RowsAffected: 0, Info: plan.UpdateInfo{ Matched: 1, Updated: 0, }, }, }, }, }, { Query: "select id, is_dirty from is_dirty", Expected: []sql.Row{ {1, 1}, }, }, }, }, { Name: "trigger before insert with subquery expressions", SetUpScript: []string{ "create table rn (id int primary key, upstream_edge_id int, downstream_edge_id int)", "create table sn (id int primary key, target_id int, source_id int)", ` create trigger rn_on_insert before insert on rn for each row begin if (select target_id from sn where id = NEW.upstream_edge_id) <> (select source_id from sn where id = NEW.downstream_edge_id) then SIGNAL SQLSTATE '45000' SET MESSAGE_TEXT = 'broken'; end if; end;`, }, Assertions: []ScriptTestAssertion{ { Query: "select id from rn", Expected: []sql.Row{}, }, }, }, { Name: "trigger before insert, multiple triggers defined", SetUpScript: []string{ "create table a (x int primary key)", "create table b (y int primary key)", "create table c (z int primary key)", "create trigger a1 before insert on a for each row insert into b values (new.x * 2)", "create trigger a2 before update on a for each row insert into b values (new.x * 3)", "create trigger a3 before delete on a for each row insert into b values (old.x * 5)", "create trigger b1 before insert on b for each row insert into c values (new.y * 7)", "create trigger b2 before update on b for each row insert into c values (new.y * 11)", "create trigger b3 before delete on b for each row insert into c values (old.y * 13)", "insert into a values (1), (2), (3)", }, Assertions: []ScriptTestAssertion{ { Query: "select x from a order by 1", Expected: []sql.Row{ {1}, {2}, {3}, }, }, { Query: "select y from b order by 1", Expected: []sql.Row{ {2}, {4}, {6}, }, }, { Query: "select z from c order by 1", Expected: []sql.Row{ {14}, {28}, {42}, }, }, }, }, { Name: "trigger with signal", SetUpScript: []string{ "create table a (x int primary key)", "create table b (y int primary key)", "create table c (z int primary key)", "insert into c values (-1)", `create trigger trig_with_signal before insert on a for each row begin declare cond_name condition for sqlstate '45000'; if new.x = 5 then signal cond_name set message_text = 'trig err'; end if; insert into b values (new.x + 1); update c set z = new.x; end;`, }, Assertions: []ScriptTestAssertion{ { Query: "insert into a values (1), (3)", Expected: []sql.Row{ {sql.OkResult{RowsAffected: 2}}, }, }, { Query: "insert into a values (5)", ExpectedErrStr: "trig err (errno 1644) (sqlstate 45000)", }, { Query: "select x from a order by 1", Expected: []sql.Row{ {1}, {3}, }, }, { Query: "select y from b order by 1", Expected: []sql.Row{ {2}, {4}, }, }, { Query: "select z from c order by 1", Expected: []sql.Row{ {3}, }, }, }, }, { Name: "show create triggers", SetUpScript: []string{ "create table a (x int primary key)", "create trigger a1 before insert on a for each row set new.x = new.x + 1", "create table b (y int primary key)", "create trigger b1 before insert on b for each row set new.y = new.y + 2", }, Assertions: []ScriptTestAssertion{ { Query: "show create trigger a1", Expected: []sql.Row{ { "a1", "", "create trigger a1 before insert on a for each row set new.x = new.x + 1", sql.Collation_Default.CharacterSet().String(), sql.Collation_Default.String(), sql.Collation_Default.String(), time.Unix(0, 0).UTC(), }, }, }, { Query: "show create trigger b1", Expected: []sql.Row{ { "b1", "", "create trigger b1 before insert on b for each row set new.y = new.y + 2", sql.Collation_Default.CharacterSet().String(), sql.Collation_Default.String(), sql.Collation_Default.String(), time.Unix(0, 0).UTC(), }, }, }, { Query: "show create trigger b2", ExpectedErr: sql.ErrTriggerDoesNotExist, }, }, }, { Name: "show triggers", SetUpScript: []string{ "create table abb (x int primary key)", "create table acc (y int primary key)", "create trigger t1 before insert on abb for each row set new.x = new.x + 1", "create trigger t2 before insert on abb for each row set new.x = new.x + 2", "create trigger t3 after insert on acc for each row insert into abb values (new.y)", "create trigger t4 before update on acc for each row set new.y = old.y + 2", }, Assertions: []ScriptTestAssertion{ { Query: "show triggers", Expected: []sql.Row{ { "t1", "INSERT", "abb", "set new.x = new.x + 1", "BEFORE", time.Unix(0, 0).UTC(), "", "", sql.Collation_Default.CharacterSet().String(), sql.Collation_Default.String(), sql.Collation_Default.String(), }, { "t2", "INSERT", "abb", "set new.x = new.x + 2", "BEFORE", time.Unix(0, 0).UTC(), "", "", sql.Collation_Default.CharacterSet().String(), sql.Collation_Default.String(), sql.Collation_Default.String(), }, { "t3", "INSERT", "acc", "insert into abb values (new.y)", "AFTER", time.Unix(0, 0).UTC(), "", "", sql.Collation_Default.CharacterSet().String(), sql.Collation_Default.String(), sql.Collation_Default.String(), }, { "t4", "UPDATE", "acc", "set new.y = old.y + 2", "BEFORE", time.Unix(0, 0).UTC(), "", "", sql.Collation_Default.CharacterSet().String(), sql.Collation_Default.String(), sql.Collation_Default.String(), }, }, }, { Query: "show triggers from mydb", Expected: []sql.Row{ { "t1", "INSERT", "abb", "set new.x = new.x + 1", "BEFORE", time.Unix(0, 0).UTC(), "", "", sql.Collation_Default.CharacterSet().String(), sql.Collation_Default.String(), sql.Collation_Default.String(), }, { "t2", "INSERT", "abb", "set new.x = new.x + 2", "BEFORE", time.Unix(0, 0).UTC(), "", "", sql.Collation_Default.CharacterSet().String(), sql.Collation_Default.String(), sql.Collation_Default.String(), }, { "t3", "INSERT", "acc", "insert into abb values (new.y)", "AFTER", time.Unix(0, 0).UTC(), "", "", sql.Collation_Default.CharacterSet().String(), sql.Collation_Default.String(), sql.Collation_Default.String(), }, { "t4", "UPDATE", "acc", "set new.y = old.y + 2", "BEFORE", time.Unix(0, 0).UTC(), "", "", sql.Collation_Default.CharacterSet().String(), sql.Collation_Default.String(), sql.Collation_Default.String(), }, }, }, { Query: "show triggers like '%cc'", Expected: []sql.Row{ { "t3", "INSERT", "acc", "insert into abb values (new.y)", "AFTER", time.Unix(0, 0).UTC(), "", "", sql.Collation_Default.CharacterSet().String(), sql.Collation_Default.String(), sql.Collation_Default.String(), }, { "t4", "UPDATE", "acc", "set new.y = old.y + 2", "BEFORE", time.Unix(0, 0).UTC(), "", "", sql.Collation_Default.CharacterSet().String(), sql.Collation_Default.String(), sql.Collation_Default.String(), }, }, }, { Query: "show triggers where `event` = 'INSERT'", Expected: []sql.Row{ { "t1", "INSERT", "abb", "set new.x = new.x + 1", "BEFORE", time.Unix(0, 0).UTC(), "", "", sql.Collation_Default.CharacterSet().String(), sql.Collation_Default.String(), sql.Collation_Default.String(), }, { "t2", "INSERT", "abb", "set new.x = new.x + 2", "BEFORE", time.Unix(0, 0).UTC(), "", "", sql.Collation_Default.CharacterSet().String(), sql.Collation_Default.String(), sql.Collation_Default.String(), }, { "t3", "INSERT", "acc", "insert into abb values (new.y)", "AFTER", time.Unix(0, 0).UTC(), "", "", sql.Collation_Default.CharacterSet().String(), sql.Collation_Default.String(), sql.Collation_Default.String(), }, }, }, { Query: "show triggers where timing = 'AFTER'", Expected: []sql.Row{ { "t3", "INSERT", "acc", "insert into abb values (new.y)", "AFTER", time.Unix(0, 0).UTC(), "", "", sql.Collation_Default.CharacterSet().String(), sql.Collation_Default.String(), sql.Collation_Default.String(), }, }, }, { Query: "show triggers where timing = 'BEFORE' and `Table` like '%bb'", Expected: []sql.Row{ { "t1", "INSERT", "abb", "set new.x = new.x + 1", "BEFORE", time.Unix(0, 0).UTC(), "", "", sql.Collation_Default.CharacterSet().String(), sql.Collation_Default.String(), sql.Collation_Default.String(), }, { "t2", "INSERT", "abb", "set new.x = new.x + 2", "BEFORE", time.Unix(0, 0).UTC(), "", "", sql.Collation_Default.CharacterSet().String(), sql.Collation_Default.String(), sql.Collation_Default.String(), }, }, }, }, }, { Name: "drop trigger", SetUpScript: []string{ "create table a (x int primary key)", "create trigger t1 before insert on a for each row set new.x = new.x * 1", "create trigger t2 before insert on a for each row follows t1 set new.x = new.x * 2", "create trigger t3 before insert on a for each row set new.x = new.x * 3", "create trigger t4 before insert on a for each row precedes t3 set new.x = new.x * 5", }, Assertions: []ScriptTestAssertion{ { Query: "drop trigger t1", ExpectedErr: sql.ErrTriggerCannotBeDropped, }, { Query: "drop trigger t3", ExpectedErr: sql.ErrTriggerCannotBeDropped, }, { Query: "drop trigger t4", Expected: []sql.Row{}, }, { Query: "drop trigger t3", Expected: []sql.Row{}, }, { Query: "drop trigger if exists t5", Expected: []sql.Row{}, }, { Query: "drop trigger t5", ExpectedErr: sql.ErrTriggerDoesNotExist, }, { Query: "select trigger_name from information_schema.triggers order by 1", Expected: []sql.Row{ {"t1"}, {"t2"}, }, }, { Query: "drop trigger if exists t2", Expected: []sql.Row{}, }, { Query: "select trigger_name from information_schema.triggers order by 1", Expected: []sql.Row{ {"t1"}, }, }, }, }, { Name: "drop table referenced in triggers", SetUpScript: []string{ "create table a (w int primary key)", "create table b (x int primary key)", "create table c (y int primary key)", "create table d (z int primary key)", "create trigger t1 before insert on a for each row set new.w = new.w", "create trigger t2 before insert on a for each row set new.w = new.w * 100", "create trigger t3 before insert on b for each row set new.x = new.x", "create trigger t4 before insert on b for each row set new.x = new.x * 100", "create trigger t5 before insert on c for each row set new.y = new.y", "create trigger t6 before insert on c for each row set new.y = new.y * 100", "create trigger t7 before insert on d for each row set new.z = new.z", "create trigger t8 before insert on d for each row set new.z = new.z * 100", }, Assertions: []ScriptTestAssertion{ { Query: "drop table a", Expected: []sql.Row{{sql.NewOkResult(0)}}, }, { Query: "select trigger_name from information_schema.triggers order by 1", Expected: []sql.Row{ {"t3"}, {"t4"}, {"t5"}, {"t6"}, {"t7"}, {"t8"}, }, }, { Query: "drop table if exists b, d, e", Expected: []sql.Row{{sql.NewOkResult(0)}}, }, { Query: "select trigger_name from information_schema.triggers order by 1", Expected: []sql.Row{ {"t5"}, {"t6"}, }, }, }, }, { Name: "drop table referenced in triggers with follows/precedes", SetUpScript: []string{ "create table a (x int primary key)", "create trigger t1 before insert on a for each row set new.x = new.x", "create trigger t2 before insert on a for each row follows t1 set new.x = new.x * 10", "create trigger t3 before insert on a for each row precedes t1 set new.x = new.x * 100", "create trigger t4 before insert on a for each row follows t3 set new.x = new.x * 1000", "create trigger t5 before insert on a for each row precedes t2 set new.x = new.x * 10000", "create trigger t6 before insert on a for each row follows t4 set new.x = new.x * 100000", "create trigger t7 before insert on a for each row precedes t1 set new.x = new.x * 1000000", "create trigger t8 before insert on a for each row follows t6 set new.x = new.x * 10000000", }, Assertions: []ScriptTestAssertion{ { Query: "drop table a", Expected: []sql.Row{{sql.NewOkResult(0)}}, }, { Query: "show triggers", Expected: []sql.Row{}, }, }, }, { Name: "triggers with subquery expressions analyze", SetUpScript: []string{ "create table a (x int primary key)", "create trigger t1 before insert on a for each row begin if NEW.x in (select 2+2 from dual) then signal SQLSTATE '45000' SET MESSAGE_TEXT = 'String field contains invalid value, like empty string, ''none'', ''null'', ''n/a'', ''nan'' etc.'; end if; end;", }, Assertions: nil, }, { Name: "insert into common sequence table (https://github.com/dolthub/dolt/issues/2534)", SetUpScript: []string{ "create table mytable (id integer PRIMARY KEY DEFAULT 0, sometext text);", "create table sequence_table (max_id integer PRIMARY KEY);", "create trigger update_position_id before insert on mytable for each row begin set new.id = (select coalesce(max(max_id),1) from sequence_table); update sequence_table set max_id = max_id + 1; end;", "insert into sequence_table values (1);", }, Assertions: []ScriptTestAssertion{ { Query: "insert into mytable () values ();", Expected: []sql.Row{{sql.NewOkResult(1)}}, }, { Query: "insert into mytable (sometext) values ('hello');", Expected: []sql.Row{{sql.NewOkResult(1)}}, }, { Query: "insert into mytable values (10, 'goodbye');", Expected: []sql.Row{{sql.NewOkResult(1)}}, }, { Query: "select * from mytable order by id", Expected: []sql.Row{ {1, nil}, {2, "hello"}, {3, "goodbye"}, }, }, }, }, { Name: "insert into common sequence table workaround", SetUpScript: []string{ "create table mytable (id integer PRIMARY KEY DEFAULT 0, sometext text);", "create table sequence_table (max_id integer PRIMARY KEY);", `create trigger update_position_id before insert on mytable for each row begin if @max_id is null then set @max_id = (select coalesce(max(max_id),1) from sequence_table); end if; set new.id = @max_id; set @max_id = @max_id + 1; update sequence_table set max_id = @max_id; end;`, "insert into sequence_table values (1);", }, Assertions: []ScriptTestAssertion{ { Query: "insert into mytable () values ();", Expected: []sql.Row{{sql.NewOkResult(1)}}, }, { Query: "insert into mytable (sometext) values ('hello');", Expected: []sql.Row{{sql.NewOkResult(1)}}, }, { Query: "insert into mytable values (10, 'goodbye');", Expected: []sql.Row{{sql.NewOkResult(1)}}, }, { Query: "insert into mytable () values (), ();", Expected: []sql.Row{{sql.NewOkResult(2)}}, }, { Query: "select * from mytable order by id", Expected: []sql.Row{ {1, nil}, {2, "hello"}, {3, "goodbye"}, {4, nil}, {5, nil}, }, }, }, }, { Name: "simple trigger with non-existent table in trigger body", SetUpScript: []string{ "create table a (x int primary key)", }, Assertions: []ScriptTestAssertion{ { Query: "create trigger insert_into_b after insert on a for each row insert into b values (new.x + 1)", Expected: []sql.Row{{sql.OkResult{}}}, }, { Query: "insert into a values (1), (3), (5)", ExpectedErr: sql.ErrTableNotFound, }, { Query: "create table b (y int primary key)", Expected: []sql.Row{{sql.OkResult{}}}, }, { Query: "insert into a values (1), (3), (5)", Expected: []sql.Row{ {sql.OkResult{RowsAffected: 3}}, }, }, { Query: "select x from a order by 1", Expected: []sql.Row{ {1}, {3}, {5}, }, }, { Query: "select y from b order by 1", Expected: []sql.Row{ {2}, {4}, {6}, }, }, }, }, { Name: "insert, update, delete triggers with non-existent table in trigger body", SetUpScript: []string{ "CREATE TABLE film (film_id smallint unsigned NOT NULL AUTO_INCREMENT, title varchar(128) NOT NULL, description text, PRIMARY KEY (film_id))", "INSERT INTO `film` VALUES (1,'ACADEMY DINOSAUR','A Epic Drama in The Canadian Rockies'),(2,'ACE GOLDFINGER','An Astounding Epistle of a Database Administrator in Ancient China');", }, Assertions: []ScriptTestAssertion{ { Query: "CREATE TRIGGER ins_film AFTER INSERT ON film FOR EACH ROW BEGIN INSERT INTO film_text (film_id, title, description) VALUES (new.film_id, new.title, new.description); END;", Expected: []sql.Row{{sql.OkResult{}}}, }, { Query: `CREATE TRIGGER upd_film AFTER UPDATE ON film FOR EACH ROW BEGIN IF (old.title != new.title) OR (old.description != new.description) OR (old.film_id != new.film_id) THEN UPDATE film_text SET title=new.title, description=new.description, film_id=new.film_id WHERE film_id=old.film_id; END IF; END;`, Expected: []sql.Row{{sql.OkResult{}}}, }, { Query: "CREATE TRIGGER del_film AFTER DELETE ON film FOR EACH ROW BEGIN DELETE FROM film_text WHERE film_id = old.film_id; END;", Expected: []sql.Row{{sql.OkResult{}}}, }, { Query: "INSERT INTO `film` VALUES (3,'ADAPTATION HOLES','An Astounding Reflection in A Baloon Factory'),(4,'AFFAIR PREJUDICE','A Fanciful Documentary in A Shark Tank')", ExpectedErr: sql.ErrTableNotFound, }, { Query: "UPDATE film SET title = 'THE ACADEMY DINOSAUR' WHERE title = 'ACADEMY DINOSAUR'", ExpectedErr: sql.ErrTableNotFound, }, { Query: "DELETE FROM film WHERE title = 'ACE GOLDFINGER'", ExpectedErr: sql.ErrTableNotFound, }, { Query: "CREATE TABLE film_text (film_id smallint NOT NULL, title varchar(255) NOT NULL, description text, PRIMARY KEY (film_id))", Expected: []sql.Row{{sql.OkResult{}}}, }, { Query: "SELECT COUNT(*) FROM film", Expected: []sql.Row{{2}}, }, { Query: "INSERT INTO `film` VALUES (3,'ADAPTATION HOLES','An Astounding Reflection in A Baloon Factory'),(4,'AFFAIR PREJUDICE','A Fanciful Documentary in A Shark Tank')", Expected: []sql.Row{{sql.OkResult{RowsAffected: 2, InsertID: 3}}}, }, { Query: "SELECT COUNT(*) FROM film", Expected: []sql.Row{{4}}, }, { Query: "SELECT COUNT(*) FROM film_text", Expected: []sql.Row{{2}}, }, { Query: "UPDATE film SET title = 'DIFFERENT MOVIE' WHERE title = 'ADAPTATION HOLES'", Expected: []sql.Row{{sql.OkResult{RowsAffected: 1, InsertID: 0, Info: plan.UpdateInfo{Matched: 1, Updated: 1, Warnings: 0}}}}, }, { Query: "SELECT COUNT(*) FROM film_text WHERE title = 'DIFFERENT MOVIE'", Expected: []sql.Row{{1}}, }, { Query: "DELETE FROM film WHERE title = 'DIFFERENT MOVIE'", Expected: []sql.Row{{sql.OkResult{RowsAffected: 1}}}, }, { Query: "SELECT COUNT(*) FROM film_text WHERE title = 'DIFFERENT MOVIE'", Expected: []sql.Row{{0}}, }, }, }, { Name: "non-existent procedure in trigger body", SetUpScript: []string{ "CREATE TABLE t0 (id INT PRIMARY KEY AUTO_INCREMENT, v1 INT, v2 TEXT);", "CREATE TABLE t1 (id INT PRIMARY KEY AUTO_INCREMENT, v1 INT, v2 TEXT);", "INSERT INTO t0 VALUES (1, 2, 'abc'), (2, 3, 'def');", }, Assertions: []ScriptTestAssertion{ { Query: "SELECT * FROM t0;", Expected: []sql.Row{{1, 2, "abc"}, {2, 3, "def"}}, }, { Query: `CREATE PROCEDURE add_entry(i INT, s TEXT) BEGIN IF i > 50 THEN SIGNAL SQLSTATE '45000' SET MESSAGE_TEXT = 'too big number'; END IF; INSERT INTO t0 (v1, v2) VALUES (i, s); END;`, Expected: []sql.Row{{sql.OkResult{}}}, }, { Query: "CREATE TRIGGER trig AFTER INSERT ON t0 FOR EACH ROW BEGIN CALL back_up(NEW.v1, NEW.v2); END;", Expected: []sql.Row{{sql.OkResult{}}}, }, { Query: "INSERT INTO t0 (v1, v2) VALUES (5, 'ggg');", ExpectedErr: sql.ErrStoredProcedureDoesNotExist, }, { Query: "CREATE PROCEDURE back_up(num INT, msg TEXT) INSERT INTO t1 (v1, v2) VALUES (num*2, msg);", Expected: []sql.Row{{sql.OkResult{}}}, }, { Query: "CALL add_entry(4, 'aaa');", Expected: []sql.Row{{sql.OkResult{RowsAffected: 1, InsertID: 1}}}, }, { Query: "SELECT * FROM t0;", Expected: []sql.Row{{1, 2, "abc"}, {2, 3, "def"}, {3, 4, "aaa"}}, }, { Query: "SELECT * FROM t1;", Expected: []sql.Row{{1, 8, "aaa"}}, }, { Query: "CALL add_entry(54, 'bbb');", ExpectedErrStr: "too big number (errno 1644) (sqlstate 45000)", }, }, }, }
var TypeWireTests = []TypeWireTest{ { Name: "TINYINT", SetUpScript: []string{ `CREATE TABLE test (pk TINYINT PRIMARY KEY, v1 TINYINT);`, `INSERT INTO test VALUES (-75, "-25"), (0, 0), (107.2, 0025), (120, -120);`, `UPDATE test SET v1 = v1 - 1 WHERE pk < 0;`, `DELETE FROM test WHERE pk > "119";`, }, Queries: []string{ `SELECT * FROM test ORDER BY pk;`, `SELECT v1, pk FROM test ORDER BY pk;`, `SELECT v1*2, pk+1 FROM test ORDER BY pk;`, }, Results: [][]sql.Row{ {{"-75", "-26"}, {"0", "0"}, {"107", "25"}}, {{"-26", "-75"}, {"0", "0"}, {"25", "107"}}, {{"-52", "-74"}, {"0", "1"}, {"50", "108"}}, }, }, { Name: "SMALLINT", SetUpScript: []string{ `CREATE TABLE test (pk SMALLINT PRIMARY KEY, v1 SMALLINT);`, `INSERT INTO test VALUES (-75, "-2531"), (0, 0), (2547.2, 03325), (9999, 9999);`, `UPDATE test SET v1 = v1 - 1 WHERE pk < 0;`, `DELETE FROM test WHERE pk >= "9999";`, }, Queries: []string{ `SELECT * FROM test ORDER BY pk;`, `SELECT v1, pk FROM test ORDER BY pk;`, `SELECT v1*2, pk+1 FROM test ORDER BY pk;`, }, Results: [][]sql.Row{ {{"-75", "-2532"}, {"0", "0"}, {"2547", "3325"}}, {{"-2532", "-75"}, {"0", "0"}, {"3325", "2547"}}, {{"-5064", "-74"}, {"0", "1"}, {"6650", "2548"}}, }, }, { Name: "MEDIUMINT", SetUpScript: []string{ `CREATE TABLE test (pk MEDIUMINT PRIMARY KEY, v1 MEDIUMINT);`, `INSERT INTO test VALUES (0, 0), (2547.2, 03325), (999999, 999999);`, `UPDATE test SET v1 = v1 - 1 WHERE pk < 0;`, `DELETE FROM test WHERE pk > "99999";`, }, Queries: []string{ `SELECT * FROM test ORDER BY pk;`, `SELECT v1, pk FROM test ORDER BY pk;`, `SELECT v1*2, pk+1 FROM test ORDER BY pk;`, }, Results: [][]sql.Row{ {{"0", "0"}, {"2547", "3325"}}, {{"0", "0"}, {"3325", "2547"}}, {{"0", "1"}, {"6650", "2548"}}, }, }, { Name: "INT", SetUpScript: []string{ `CREATE TABLE test (pk INT PRIMARY KEY, v1 INT);`, `INSERT INTO test VALUES (-75, "-2531"), (0, 0), (2547.2, 03325), (999999, 999999);`, `UPDATE test SET v1 = v1 - 1 WHERE pk < 0;`, `DELETE FROM test WHERE pk > "99999";`, }, Queries: []string{ `SELECT * FROM test ORDER BY pk;`, `SELECT v1, pk FROM test ORDER BY pk;`, `SELECT v1*2, pk+1 FROM test ORDER BY pk;`, }, Results: [][]sql.Row{ {{"-75", "-2532"}, {"0", "0"}, {"2547", "3325"}}, {{"-2532", "-75"}, {"0", "0"}, {"3325", "2547"}}, {{"-5064", "-74"}, {"0", "1"}, {"6650", "2548"}}, }, }, { Name: "BIGINT", SetUpScript: []string{ `CREATE TABLE test (pk BIGINT PRIMARY KEY, v1 BIGINT);`, `INSERT INTO test VALUES (-75, "-2531"), (0, 0), (2547.2, 03325), (999999, 999999);`, `UPDATE test SET v1 = v1 - 1 WHERE pk < 0;`, `DELETE FROM test WHERE pk > "99999";`, }, Queries: []string{ `SELECT * FROM test ORDER BY pk;`, `SELECT v1, pk FROM test ORDER BY pk;`, `SELECT v1*2, pk+1 FROM test ORDER BY pk;`, }, Results: [][]sql.Row{ {{"-75", "-2532"}, {"0", "0"}, {"2547", "3325"}}, {{"-2532", "-75"}, {"0", "0"}, {"3325", "2547"}}, {{"-5064", "-74"}, {"0", "1"}, {"6650", "2548"}}, }, }, { Name: "TINYINT UNSIGNED", SetUpScript: []string{ `CREATE TABLE test (pk TINYINT UNSIGNED PRIMARY KEY, v1 TINYINT UNSIGNED);`, `INSERT INTO test VALUES (0, 0), (25, "26"), (32.1, 0126), (255, 255);`, `UPDATE test SET v1 = v1 - 1 WHERE pk > 0 AND pk < 30;`, `DELETE FROM test WHERE pk >= "255";`, }, Queries: []string{ `SELECT * FROM test ORDER BY pk;`, `SELECT v1, pk FROM test ORDER BY pk;`, `SELECT v1*2, pk+1 FROM test ORDER BY pk;`, }, Results: [][]sql.Row{ {{"0", "0"}, {"25", "25"}, {"32", "126"}}, {{"0", "0"}, {"25", "25"}, {"126", "32"}}, {{"0", "1"}, {"50", "26"}, {"252", "33"}}, }, }, { Name: "SMALLINT UNSIGNED", SetUpScript: []string{ `CREATE TABLE test (pk SMALLINT UNSIGNED PRIMARY KEY, v1 SMALLINT UNSIGNED);`, `INSERT INTO test VALUES (0, 0), (25, "2531"), (2547.2, 03325), (9999, 9999);`, `UPDATE test SET v1 = v1 - 1 WHERE pk > 0 AND pk < 100;`, `DELETE FROM test WHERE pk >= "9999";`, }, Queries: []string{ `SELECT * FROM test ORDER BY pk;`, `SELECT v1, pk FROM test ORDER BY pk;`, `SELECT v1*2, pk+1 FROM test ORDER BY pk;`, }, Results: [][]sql.Row{ {{"0", "0"}, {"25", "2530"}, {"2547", "3325"}}, {{"0", "0"}, {"2530", "25"}, {"3325", "2547"}}, {{"0", "1"}, {"5060", "26"}, {"6650", "2548"}}, }, }, { Name: "MEDIUMINT UNSIGNED", SetUpScript: []string{ `CREATE TABLE test (pk MEDIUMINT UNSIGNED PRIMARY KEY, v1 MEDIUMINT UNSIGNED);`, `INSERT INTO test VALUES (75, "2531"), (0, 0), (2547.2, 03325), (999999, 999999);`, `UPDATE test SET v1 = v1 + 1 WHERE pk < 100;`, `DELETE FROM test WHERE pk > "99999";`, }, Queries: []string{ `SELECT * FROM test ORDER BY pk;`, `SELECT v1, pk FROM test ORDER BY pk;`, `SELECT v1*2, pk+1 FROM test ORDER BY pk;`, }, Results: [][]sql.Row{ {{"0", "1"}, {"75", "2532"}, {"2547", "3325"}}, {{"1", "0"}, {"2532", "75"}, {"3325", "2547"}}, {{"2", "1"}, {"5064", "76"}, {"6650", "2548"}}, }, }, { Name: "INT UNSIGNED", SetUpScript: []string{ `CREATE TABLE test (pk INT UNSIGNED PRIMARY KEY, v1 INT UNSIGNED);`, `INSERT INTO test VALUES (75, "2531"), (0, 0), (2547.2, 03325), (999999, 999999);`, `UPDATE test SET v1 = v1 + 1 WHERE pk < 100;`, `DELETE FROM test WHERE pk > "99999";`, }, Queries: []string{ `SELECT * FROM test ORDER BY pk;`, `SELECT v1, pk FROM test ORDER BY pk;`, `SELECT v1*2, pk+1 FROM test ORDER BY pk;`, }, Results: [][]sql.Row{ {{"0", "1"}, {"75", "2532"}, {"2547", "3325"}}, {{"1", "0"}, {"2532", "75"}, {"3325", "2547"}}, {{"2", "1"}, {"5064", "76"}, {"6650", "2548"}}, }, }, { Name: "BIGINT UNSIGNED", SetUpScript: []string{ `CREATE TABLE test (pk BIGINT UNSIGNED PRIMARY KEY, v1 BIGINT UNSIGNED);`, `INSERT INTO test VALUES (75, "2531"), (0, 0), (2547.2, 03325), (999999, 999999);`, `UPDATE test SET v1 = v1 + 1 WHERE pk < 100;`, `DELETE FROM test WHERE pk > "99999";`, }, Queries: []string{ `SELECT * FROM test ORDER BY pk;`, `SELECT v1, pk FROM test ORDER BY pk;`, `SELECT v1*2, pk+1 FROM test ORDER BY pk;`, }, Results: [][]sql.Row{ {{"0", "1"}, {"75", "2532"}, {"2547", "3325"}}, {{"1", "0"}, {"2532", "75"}, {"3325", "2547"}}, {{"2", "1"}, {"5064", "76"}, {"6650", "2548"}}, }, }, { Name: "FLOAT", SetUpScript: []string{ `CREATE TABLE test (pk FLOAT PRIMARY KEY, v1 FLOAT);`, `INSERT INTO test VALUES (-75.11, "-2531"), (0, 0), ("2547.2", 03325), (999999, 999999);`, `UPDATE test SET v1 = v1 - 1 WHERE pk < 0;`, `DELETE FROM test WHERE pk > "99999";`, }, Queries: []string{ `SELECT * FROM test ORDER BY pk;`, `SELECT v1, pk FROM test ORDER BY pk;`, `SELECT v1*2, pk+1 FROM test ORDER BY pk;`, }, Results: [][]sql.Row{ {{"-75.11", "-2532"}, {"0", "0"}, {"2547.2", "3325"}}, {{"-2532", "-75.11"}, {"0", "0"}, {"3325", "2547.2"}}, {{"-5064", "-74.11000061035156"}, {"0", "1"}, {"6650", "2548.199951171875"}}, }, }, { Name: "DOUBLE", SetUpScript: []string{ `CREATE TABLE test (pk DOUBLE PRIMARY KEY, v1 DOUBLE);`, `INSERT INTO test VALUES (-75.11, "-2531"), (0, 0), ("2547.2", 03325), (999999, 999999);`, `UPDATE test SET v1 = v1 - 1 WHERE pk < 0;`, `DELETE FROM test WHERE pk > "99999";`, }, Queries: []string{ `SELECT * FROM test ORDER BY pk;`, `SELECT v1, pk FROM test ORDER BY pk;`, `SELECT v1*2, pk+1 FROM test ORDER BY pk;`, }, Results: [][]sql.Row{ {{"-75.11", "-2532"}, {"0", "0"}, {"2547.2", "3325"}}, {{"-2532", "-75.11"}, {"0", "0"}, {"3325", "2547.2"}}, {{"-5064", "-74.11"}, {"0", "1"}, {"6650", "2548.2"}}, }, }, { Name: "DECIMAL", SetUpScript: []string{ `CREATE TABLE test (pk DECIMAL(5,0) PRIMARY KEY, v1 DECIMAL(25,5));`, `INSERT INTO test VALUES (-75, "-2531.356"), (0, 0), (2547.2, 03325), (99999, 999999);`, `UPDATE test SET v1 = v1 - 1 WHERE pk < 0;`, `DELETE FROM test WHERE pk >= "99999";`, }, Queries: []string{ `SELECT * FROM test ORDER BY pk;`, `SELECT v1, pk FROM test ORDER BY pk;`, `SELECT v1*2, pk+1 FROM test ORDER BY pk;`, }, Results: [][]sql.Row{ {{"-75", "-2532.35600"}, {"0", "0.00000"}, {"2547", "3325.00000"}}, {{"-2532.35600", "-75"}, {"0.00000", "0"}, {"3325.00000", "2547"}}, {{"-5064.712", "-74"}, {"0", "1"}, {"6650", "2548"}}, }, }, { Name: "BIT", SetUpScript: []string{ `CREATE TABLE test (pk BIT(55) PRIMARY KEY, v1 BIT(1), v2 BIT(24));`, `INSERT INTO test VALUES (75, 0, "21"), (0, 0, 0), (2547.2, 1, 03325), (999999, 1, 999999);`, `UPDATE test SET v2 = v2 - 1 WHERE pk > 0 AND pk < 100;`, `DELETE FROM test WHERE pk > 99999;`, }, Queries: []string{ `SELECT * FROM test ORDER BY pk;`, `SELECT v2, v1, pk FROM test ORDER BY pk;`, `SELECT v1*1, pk/10, v2+1 FROM test ORDER BY pk;`, }, Results: [][]sql.Row{ {{"\x00\x00\x00\x00\x00\x00\x00", "\x00", "\x00\x00\x00"}, {"\x00\x00\x00\x00\x00\x00K", "\x00", "\x0020"}, {"\x00\x00\x00\x00\x00\t\xf3", "�", "\x00\xfd"}}, {{"\x00\x00\x00", "\x00", "\x00\x00\x00\x00\x00\x00\x00"}, {"\x0020", "\x00", "\x00\x00\x00\x00\x00\x00K"}, {"\x00\xfd", "�", "\x00\x00\x00\x00\x00\t\xf3"}}, {{"0", "0", "1"}, {"0", "7.5", "12849"}, {"1", "254.7", "3326"}}, }, }, { Name: "YEAR", SetUpScript: []string{ `CREATE TABLE test (pk YEAR PRIMARY KEY, v1 YEAR);`, `INSERT INTO test VALUES (1901, 1901), (1950, "1950"), (1979.2, 01986), (2122, 2122);`, `UPDATE test SET v1 = v1 + 1 WHERE pk < 1975;`, `DELETE FROM test WHERE pk > "2100";`, }, Queries: []string{ `SELECT * FROM test ORDER BY pk;`, `SELECT v1, pk FROM test ORDER BY pk;`, `SELECT v1+3, pk+2 FROM test ORDER BY pk;`, }, Results: [][]sql.Row{ {{"1901", "1902"}, {"1950", "1951"}, {"1979", "1986"}}, {{"1902", "1901"}, {"1951", "1950"}, {"1986", "1979"}}, {{"1905", "1903"}, {"1954", "1952"}, {"1989", "1981"}}, }, }, { Name: "TIMESTAMP", SetUpScript: []string{ `CREATE TABLE test (pk TIMESTAMP PRIMARY KEY, v1 TIMESTAMP);`, `INSERT INTO test VALUES ("1980-04-12 12:02:11", "1986-08-02 17:04:22"), ("1999-11-28 13:06:33", "2022-01-14 15:08:44"), ("2020-05-06 18:10:55", "1975-09-15 11:12:16");`, `UPDATE test SET v1 = "2000-01-01 00:00:00" WHERE pk < "1990-01-01 00:00:00";`, `DELETE FROM test WHERE pk > "2015-01-01 00:00:00";`, }, Queries: []string{ `SELECT * FROM test ORDER BY pk;`, `SELECT pk, v1 FROM test ORDER BY pk;`, `SELECT v1, pk FROM test ORDER BY pk;`, }, Results: [][]sql.Row{ {{"1980-04-12 12:02:11", "2000-01-01 00:00:00"}, {"1999-11-28 13:06:33", "2022-01-14 15:08:44"}}, {{"1980-04-12 12:02:11", "2000-01-01 00:00:00"}, {"1999-11-28 13:06:33", "2022-01-14 15:08:44"}}, {{"2000-01-01 00:00:00", "1980-04-12 12:02:11"}, {"2022-01-14 15:08:44", "1999-11-28 13:06:33"}}, }, }, { Name: "DATETIME", SetUpScript: []string{ `CREATE TABLE test (pk DATETIME PRIMARY KEY, v1 DATETIME);`, `INSERT INTO test VALUES ("1000-04-12 12:02:11", "1986-08-02 17:04:22"), ("1999-11-28 13:06:33", "2022-01-14 15:08:44"), ("5020-05-06 18:10:55", "1975-09-15 11:12:16");`, `UPDATE test SET v1 = "2000-01-01 00:00:00" WHERE pk < "1990-01-01 00:00:00";`, `DELETE FROM test WHERE pk > "5000-01-01 00:00:00";`, }, Queries: []string{ `SELECT * FROM test ORDER BY pk;`, `SELECT pk, v1 FROM test ORDER BY pk;`, `SELECT v1, pk FROM test ORDER BY pk;`, }, Results: [][]sql.Row{ {{"1000-04-12 12:02:11", "2000-01-01 00:00:00"}, {"1999-11-28 13:06:33", "2022-01-14 15:08:44"}}, {{"1000-04-12 12:02:11", "2000-01-01 00:00:00"}, {"1999-11-28 13:06:33", "2022-01-14 15:08:44"}}, {{"2000-01-01 00:00:00", "1000-04-12 12:02:11"}, {"2022-01-14 15:08:44", "1999-11-28 13:06:33"}}, }, }, { Name: "DATE", SetUpScript: []string{ `CREATE TABLE test (pk DATE PRIMARY KEY, v1 DATE);`, `INSERT INTO test VALUES ("1000-04-12", "1986-08-02"), ("1999-11-28", "2022-01-14"), ("5020-05-06", "1975-09-15");`, `UPDATE test SET v1 = "2000-01-01" WHERE pk < "1990-01-01";`, `DELETE FROM test WHERE pk > "5000-01-01";`, }, Queries: []string{ `SELECT * FROM test ORDER BY pk;`, `SELECT pk, v1 FROM test ORDER BY pk;`, `SELECT v1, pk FROM test ORDER BY pk;`, }, Results: [][]sql.Row{ {{"1000-04-12", "2000-01-01"}, {"1999-11-28", "2022-01-14"}}, {{"1000-04-12", "2000-01-01"}, {"1999-11-28", "2022-01-14"}}, {{"2000-01-01", "1000-04-12"}, {"2022-01-14", "1999-11-28"}}, }, }, { Name: "TIME", SetUpScript: []string{ `CREATE TABLE test (pk TIME PRIMARY KEY, v1 TIME);`, `INSERT INTO test VALUES ("-800:00:00", "-20:21:22"), ("00:00:00", "00:00:00"), ("10:26:57", "30:53:14"), ("700:23:51", "300:25:52");`, `UPDATE test SET v1 = "-120:12:20" WHERE pk < "00:00:00";`, `DELETE FROM test WHERE pk > "600:00:00";`, }, Queries: []string{ `SELECT * FROM test ORDER BY pk;`, `SELECT pk, v1 FROM test ORDER BY pk;`, `SELECT v1, pk FROM test ORDER BY pk;`, }, Results: [][]sql.Row{ {{"-800:00:00", "-120:12:20"}, {"00:00:00", "00:00:00"}, {"10:26:57", "30:53:14"}}, {{"-800:00:00", "-120:12:20"}, {"00:00:00", "00:00:00"}, {"10:26:57", "30:53:14"}}, {{"-120:12:20", "-800:00:00"}, {"00:00:00", "00:00:00"}, {"30:53:14", "10:26:57"}}, }, }, { Name: "CHAR", SetUpScript: []string{ `CREATE TABLE test (pk BIGINT PRIMARY KEY, v1 CHAR(5), v2 CHAR(10));`, `INSERT INTO test VALUES (1, "abc", "def"), (2, "c-a", "123"), (3, "__2", 456), (4, "?hi?", "\\n");`, `UPDATE test SET v1 = "a-c" WHERE pk = 2;`, `DELETE FROM test WHERE pk = 4;`, }, Queries: []string{ `SELECT * FROM test ORDER BY pk;`, `SELECT pk, v2, v1 FROM test ORDER BY pk;`, `SELECT CONCAT(v1, "r"), pk, v2 FROM test ORDER BY pk;`, }, Results: [][]sql.Row{ {{"1", "abc", "def"}, {"2", "a-c", "123"}, {"3", "__2", "456"}}, {{"1", "def", "abc"}, {"2", "123", "a-c"}, {"3", "456", "__2"}}, {{"abcr", "1", "def"}, {"a-cr", "2", "123"}, {"__2r", "3", "456"}}, }, }, { Name: "VARCHAR", SetUpScript: []string{ `CREATE TABLE test (pk BIGINT PRIMARY KEY, v1 VARCHAR(5), v2 VARCHAR(10));`, `INSERT INTO test VALUES (1, "abc", "def"), (2, "c-a", "123"), (3, "__2", 456), (4, "?hi?", "\\n");`, `UPDATE test SET v1 = CONCAT(v1, "x") WHERE pk = 2;`, `DELETE FROM test WHERE pk = 4;`, }, Queries: []string{ `SELECT * FROM test ORDER BY pk;`, `SELECT pk, v2, v1 FROM test ORDER BY pk;`, `SELECT CONCAT(v1, "r"), pk, v2 FROM test ORDER BY pk;`, }, Results: [][]sql.Row{ {{"1", "abc", "def"}, {"2", "c-ax", "123"}, {"3", "__2", "456"}}, {{"1", "def", "abc"}, {"2", "123", "c-ax"}, {"3", "456", "__2"}}, {{"abcr", "1", "def"}, {"c-axr", "2", "123"}, {"__2r", "3", "456"}}, }, }, { Name: "BINARY", SetUpScript: []string{ `CREATE TABLE test (pk BIGINT PRIMARY KEY, v1 BINARY(5), v2 BINARY(10));`, `INSERT INTO test VALUES (1, "abc", "def"), (2, "c-a", "123"), (3, "__2", 456), (4, "?hi?", "\\n");`, `UPDATE test SET v1 = "a-c" WHERE pk = 2;`, `DELETE FROM test WHERE pk = 4;`, }, Queries: []string{ `SELECT * FROM test ORDER BY pk;`, `SELECT pk, v2, v1 FROM test ORDER BY pk;`, `SELECT CONCAT(v1, "r"), pk, v2 FROM test ORDER BY pk;`, }, Results: [][]sql.Row{ {{"1", "abc\x00\x00", "def\x00\x00\x00\x00\x00\x00\x00"}, {"2", "a-c\x00\x00", "123\x00\x00\x00\x00\x00\x00\x00"}, {"3", "__2\x00\x00", "456\x00\x00\x00\x00\x00\x00\x00"}}, {{"1", "def\x00\x00\x00\x00\x00\x00\x00", "abc\x00\x00"}, {"2", "123\x00\x00\x00\x00\x00\x00\x00", "a-c\x00\x00"}, {"3", "456\x00\x00\x00\x00\x00\x00\x00", "__2\x00\x00"}}, {{"abc\x00\x00r", "1", "def\x00\x00\x00\x00\x00\x00\x00"}, {"a-c\x00\x00r", "2", "123\x00\x00\x00\x00\x00\x00\x00"}, {"__2\x00\x00r", "3", "456\x00\x00\x00\x00\x00\x00\x00"}}, }, }, { Name: "VARBINARY", SetUpScript: []string{ `CREATE TABLE test (pk BIGINT PRIMARY KEY, v1 VARBINARY(5), v2 VARBINARY(10));`, `INSERT INTO test VALUES (1, "abc", "def"), (2, "c-a", "123"), (3, "__2", 456), (4, "?hi?", "\\n");`, `UPDATE test SET v1 = CONCAT(v1, "x") WHERE pk = 2;`, `DELETE FROM test WHERE pk = 4;`, }, Queries: []string{ `SELECT * FROM test ORDER BY pk;`, `SELECT pk, v2, v1 FROM test ORDER BY pk;`, `SELECT CONCAT(v1, "r"), pk, v2 FROM test ORDER BY pk;`, }, Results: [][]sql.Row{ {{"1", "abc", "def"}, {"2", "c-ax", "123"}, {"3", "__2", "456"}}, {{"1", "def", "abc"}, {"2", "123", "c-ax"}, {"3", "456", "__2"}}, {{"abcr", "1", "def"}, {"c-axr", "2", "123"}, {"__2r", "3", "456"}}, }, }, { Name: "TINYTEXT", SetUpScript: []string{ `CREATE TABLE test (pk BIGINT PRIMARY KEY, v1 TINYTEXT, v2 TINYTEXT);`, `INSERT INTO test VALUES (1, "abc", "def"), (2, "c-a", "123"), (3, "__2", 456), (4, "?hi?", "\\n");`, `UPDATE test SET v1 = CONCAT(v1, "x") WHERE pk = 2;`, `DELETE FROM test WHERE pk = 4;`, }, Queries: []string{ `SELECT * FROM test ORDER BY pk;`, `SELECT pk, v2, v1 FROM test ORDER BY pk;`, `SELECT CONCAT(v1, "r"), pk, v2 FROM test ORDER BY pk;`, }, Results: [][]sql.Row{ {{"1", "abc", "def"}, {"2", "c-ax", "123"}, {"3", "__2", "456"}}, {{"1", "def", "abc"}, {"2", "123", "c-ax"}, {"3", "456", "__2"}}, {{"abcr", "1", "def"}, {"c-axr", "2", "123"}, {"__2r", "3", "456"}}, }, }, { Name: "TEXT", SetUpScript: []string{ `CREATE TABLE test (pk BIGINT PRIMARY KEY, v1 TEXT, v2 TEXT);`, `INSERT INTO test VALUES (1, "abc", "def"), (2, "c-a", "123"), (3, "__2", 456), (4, "?hi?", "\\n");`, `UPDATE test SET v1 = CONCAT(v1, "x") WHERE pk = 2;`, `DELETE FROM test WHERE pk = 4;`, }, Queries: []string{ `SELECT * FROM test ORDER BY pk;`, `SELECT pk, v2, v1 FROM test ORDER BY pk;`, `SELECT CONCAT(v1, "r"), pk, v2 FROM test ORDER BY pk;`, }, Results: [][]sql.Row{ {{"1", "abc", "def"}, {"2", "c-ax", "123"}, {"3", "__2", "456"}}, {{"1", "def", "abc"}, {"2", "123", "c-ax"}, {"3", "456", "__2"}}, {{"abcr", "1", "def"}, {"c-axr", "2", "123"}, {"__2r", "3", "456"}}, }, }, { Name: "MEDIUMTEXT", SetUpScript: []string{ `CREATE TABLE test (pk BIGINT PRIMARY KEY, v1 MEDIUMTEXT, v2 MEDIUMTEXT);`, `INSERT INTO test VALUES (1, "abc", "def"), (2, "c-a", "123"), (3, "__2", 456), (4, "?hi?", "\\n");`, `UPDATE test SET v1 = CONCAT(v1, "x") WHERE pk = 2;`, `DELETE FROM test WHERE pk = 4;`, }, Queries: []string{ `SELECT * FROM test ORDER BY pk;`, `SELECT pk, v2, v1 FROM test ORDER BY pk;`, `SELECT CONCAT(v1, "r"), pk, v2 FROM test ORDER BY pk;`, }, Results: [][]sql.Row{ {{"1", "abc", "def"}, {"2", "c-ax", "123"}, {"3", "__2", "456"}}, {{"1", "def", "abc"}, {"2", "123", "c-ax"}, {"3", "456", "__2"}}, {{"abcr", "1", "def"}, {"c-axr", "2", "123"}, {"__2r", "3", "456"}}, }, }, { Name: "LONGTEXT", SetUpScript: []string{ `CREATE TABLE test (pk BIGINT PRIMARY KEY, v1 LONGTEXT, v2 LONGTEXT);`, `INSERT INTO test VALUES (1, "abc", "def"), (2, "c-a", "123"), (3, "__2", 456), (4, "?hi?", "\\n");`, `UPDATE test SET v1 = CONCAT(v1, "x") WHERE pk = 2;`, `DELETE FROM test WHERE pk = 4;`, }, Queries: []string{ `SELECT * FROM test ORDER BY pk;`, `SELECT pk, v2, v1 FROM test ORDER BY pk;`, `SELECT CONCAT(v1, "r"), pk, v2 FROM test ORDER BY pk;`, }, Results: [][]sql.Row{ {{"1", "abc", "def"}, {"2", "c-ax", "123"}, {"3", "__2", "456"}}, {{"1", "def", "abc"}, {"2", "123", "c-ax"}, {"3", "456", "__2"}}, {{"abcr", "1", "def"}, {"c-axr", "2", "123"}, {"__2r", "3", "456"}}, }, }, { Name: "TINYBLOB", SetUpScript: []string{ `CREATE TABLE test (pk BIGINT PRIMARY KEY, v1 TINYBLOB, v2 TINYBLOB);`, `INSERT INTO test VALUES (1, "abc", "def"), (2, "c-a", "123"), (3, "__2", 456), (4, "?hi?", "\\n");`, `UPDATE test SET v1 = CONCAT(v1, "x") WHERE pk = 2;`, `DELETE FROM test WHERE pk = 4;`, }, Queries: []string{ `SELECT * FROM test ORDER BY pk;`, `SELECT pk, v2, v1 FROM test ORDER BY pk;`, `SELECT CONCAT(v1, "r"), pk, v2 FROM test ORDER BY pk;`, }, Results: [][]sql.Row{ {{"1", "abc", "def"}, {"2", "c-ax", "123"}, {"3", "__2", "456"}}, {{"1", "def", "abc"}, {"2", "123", "c-ax"}, {"3", "456", "__2"}}, {{"abcr", "1", "def"}, {"c-axr", "2", "123"}, {"__2r", "3", "456"}}, }, }, { Name: "BLOB", SetUpScript: []string{ `CREATE TABLE test (pk BIGINT PRIMARY KEY, v1 BLOB, v2 BLOB);`, `INSERT INTO test VALUES (1, "abc", "def"), (2, "c-a", "123"), (3, "__2", 456), (4, "?hi?", "\\n");`, `UPDATE test SET v1 = CONCAT(v1, "x") WHERE pk = 2;`, `DELETE FROM test WHERE pk = 4;`, }, Queries: []string{ `SELECT * FROM test ORDER BY pk;`, `SELECT pk, v2, v1 FROM test ORDER BY pk;`, `SELECT CONCAT(v1, "r"), pk, v2 FROM test ORDER BY pk;`, }, Results: [][]sql.Row{ {{"1", "abc", "def"}, {"2", "c-ax", "123"}, {"3", "__2", "456"}}, {{"1", "def", "abc"}, {"2", "123", "c-ax"}, {"3", "456", "__2"}}, {{"abcr", "1", "def"}, {"c-axr", "2", "123"}, {"__2r", "3", "456"}}, }, }, { Name: "MEDIUMBLOB", SetUpScript: []string{ `CREATE TABLE test (pk BIGINT PRIMARY KEY, v1 MEDIUMBLOB, v2 MEDIUMBLOB);`, `INSERT INTO test VALUES (1, "abc", "def"), (2, "c-a", "123"), (3, "__2", 456), (4, "?hi?", "\\n");`, `UPDATE test SET v1 = CONCAT(v1, "x") WHERE pk = 2;`, `DELETE FROM test WHERE pk = 4;`, }, Queries: []string{ `SELECT * FROM test ORDER BY pk;`, `SELECT pk, v2, v1 FROM test ORDER BY pk;`, `SELECT CONCAT(v1, "r"), pk, v2 FROM test ORDER BY pk;`, }, Results: [][]sql.Row{ {{"1", "abc", "def"}, {"2", "c-ax", "123"}, {"3", "__2", "456"}}, {{"1", "def", "abc"}, {"2", "123", "c-ax"}, {"3", "456", "__2"}}, {{"abcr", "1", "def"}, {"c-axr", "2", "123"}, {"__2r", "3", "456"}}, }, }, { Name: "LONGBLOB", SetUpScript: []string{ `CREATE TABLE test (pk BIGINT PRIMARY KEY, v1 LONGBLOB, v2 LONGBLOB);`, `INSERT INTO test VALUES (1, "abc", "def"), (2, "c-a", "123"), (3, "__2", 456), (4, "?hi?", "\\n");`, `UPDATE test SET v1 = CONCAT(v1, "x") WHERE pk = 2;`, `DELETE FROM test WHERE pk = 4;`, }, Queries: []string{ `SELECT * FROM test ORDER BY pk;`, `SELECT pk, v2, v1 FROM test ORDER BY pk;`, `SELECT CONCAT(v1, "r"), pk, v2 FROM test ORDER BY pk;`, }, Results: [][]sql.Row{ {{"1", "abc", "def"}, {"2", "c-ax", "123"}, {"3", "__2", "456"}}, {{"1", "def", "abc"}, {"2", "123", "c-ax"}, {"3", "456", "__2"}}, {{"abcr", "1", "def"}, {"c-axr", "2", "123"}, {"__2r", "3", "456"}}, }, }, { Name: "ENUM", SetUpScript: []string{ `CREATE TABLE test (pk ENUM("a","b","c") PRIMARY KEY, v1 ENUM("x","y","z"));`, `INSERT INTO test VALUES (1, 1), ("b", "y"), (3, "z");`, `UPDATE test SET v1 = "x" WHERE pk = 2;`, `DELETE FROM test WHERE pk > 2;`, }, Queries: []string{ `SELECT * FROM test ORDER BY pk;`, `SELECT pk, v1 FROM test ORDER BY pk;`, `SELECT v1, pk FROM test ORDER BY pk;`, }, Results: [][]sql.Row{ {{"a", "x"}, {"b", "x"}}, {{"a", "x"}, {"b", "x"}}, {{"x", "a"}, {"x", "b"}}, }, }, { Name: "SET", SetUpScript: []string{ `CREATE TABLE test (pk SET("a","b","c") PRIMARY KEY, v1 SET("w","x","y","z"));`, `INSERT INTO test VALUES (0, 1), ("b", "y"), ("b,c", "z,z"), ("a,c,b", 10);`, `UPDATE test SET v1 = "y,x,w" WHERE pk >= 4`, `DELETE FROM test WHERE pk > "b,c";`, }, Queries: []string{ `SELECT * FROM test ORDER BY pk;`, `SELECT pk, v1 FROM test ORDER BY pk;`, `SELECT v1, pk FROM test ORDER BY pk;`, }, Results: [][]sql.Row{ {{"", "w"}, {"b", "y"}, {"b,c", "w,x,y"}}, {{"", "w"}, {"b", "y"}, {"b,c", "w,x,y"}}, {{"w", ""}, {"y", "b"}, {"w,x,y", "b,c"}}, }, }, { Name: "JSON", SetUpScript: []string{ `CREATE TABLE test (pk BIGINT PRIMARY KEY, v1 JSON);`, `INSERT INTO test VALUES (1, '{"key1": {"key": "value"}}'), (2, '{"key1": "value1", "key2": "value2"}'), (3, '{"key1": {"key": [2,3]}}');`, `UPDATE test SET v1 = '["a", 1]' WHERE pk = 1;`, `DELETE FROM test WHERE pk = 3;`, }, Queries: []string{ `SELECT * FROM test ORDER BY pk;`, `SELECT v1, pk FROM test ORDER BY pk;`, `SELECT pk, JSON_ARRAYAGG(v1) FROM (SELECT * FROM test ORDER BY pk) as sub GROUP BY v1 ORDER BY pk;`, }, Results: [][]sql.Row{ {{"1", "[\"a\",1]"}, {"2", "{\"key1\":\"value1\",\"key2\":\"value2\"}"}}, {{"[\"a\",1]", "1"}, {"{\"key1\":\"value1\",\"key2\":\"value2\"}", "2"}}, {{"1", "[[\"a\",1]]"}, {"2", "[{\"key1\":\"value1\",\"key2\":\"value2\"}]"}}, }, }, }
TypeWireTests are used to ensure that types are properly represented over the wire (vs being directly returned from the engine).
var UpdateErrorScripts = []ScriptTest{ { Name: "try updating string that is too long", SetUpScript: []string{ "create table bad (s varchar(9))", "insert into bad values ('good')", }, Query: "update bad set s = '1234567890'", ExpectedErr: sql.ErrLengthBeyondLimit, }, }
var UpdateErrorTests = []QueryErrorTest{ { Query: `UPDATE keyless INNER JOIN one_pk on keyless.c0 = one_pk.pk SET keyless.c0 = keyless.c0 + 1`, ExpectedErr: sql.ErrUnsupportedFeature, }, { Query: `UPDATE people set height_inches = null where height_inches < 100`, ExpectedErr: sql.ErrInsertIntoNonNullableProvidedNull, }, }
var UpdateTests = []WriteQueryTest{ { WriteQuery: "UPDATE mytable SET s = 'updated';", ExpectedWriteResult: []sql.Row{{newUpdateResult(3, 3)}}, SelectQuery: "SELECT * FROM mytable;", ExpectedSelect: []sql.Row{{int64(1), "updated"}, {int64(2), "updated"}, {int64(3), "updated"}}, }, { WriteQuery: "UPDATE mytable SET S = 'updated';", ExpectedWriteResult: []sql.Row{{newUpdateResult(3, 3)}}, SelectQuery: "SELECT * FROM mytable;", ExpectedSelect: []sql.Row{{int64(1), "updated"}, {int64(2), "updated"}, {int64(3), "updated"}}, }, { WriteQuery: "UPDATE mytable SET s = 'updated' WHERE i > 9999;", ExpectedWriteResult: []sql.Row{{newUpdateResult(0, 0)}}, SelectQuery: "SELECT * FROM mytable;", ExpectedSelect: []sql.Row{{int64(1), "first row"}, {int64(2), "second row"}, {int64(3), "third row"}}, }, { WriteQuery: "UPDATE mytable SET s = 'updated' WHERE i = 1;", ExpectedWriteResult: []sql.Row{{newUpdateResult(1, 1)}}, SelectQuery: "SELECT * FROM mytable;", ExpectedSelect: []sql.Row{{int64(1), "updated"}, {int64(2), "second row"}, {int64(3), "third row"}}, }, { WriteQuery: "UPDATE mytable SET s = 'updated' WHERE i <> 9999;", ExpectedWriteResult: []sql.Row{{newUpdateResult(3, 3)}}, SelectQuery: "SELECT * FROM mytable;", ExpectedSelect: []sql.Row{{int64(1), "updated"}, {int64(2), "updated"}, {int64(3), "updated"}}, }, { WriteQuery: "UPDATE floattable SET f32 = f32 + f32, f64 = f32 * f64 WHERE i = 2;", ExpectedWriteResult: []sql.Row{{newUpdateResult(1, 1)}}, SelectQuery: "SELECT * FROM floattable WHERE i = 2;", ExpectedSelect: []sql.Row{{int64(2), float32(3.0), float64(4.5)}}, }, { WriteQuery: "UPDATE floattable SET f32 = 5, f32 = 4 WHERE i = 1;", ExpectedWriteResult: []sql.Row{{newUpdateResult(1, 1)}}, SelectQuery: "SELECT f32 FROM floattable WHERE i = 1;", ExpectedSelect: []sql.Row{{float32(4.0)}}, }, { WriteQuery: "UPDATE mytable SET s = 'first row' WHERE i = 1;", ExpectedWriteResult: []sql.Row{{newUpdateResult(1, 0)}}, SelectQuery: "SELECT * FROM mytable;", ExpectedSelect: []sql.Row{{int64(1), "first row"}, {int64(2), "second row"}, {int64(3), "third row"}}, }, { WriteQuery: "UPDATE niltable SET b = NULL WHERE f IS NULL;", ExpectedWriteResult: []sql.Row{{newUpdateResult(3, 2)}}, SelectQuery: "SELECT i,b FROM niltable WHERE f IS NULL;", ExpectedSelect: []sql.Row{{int64(1), nil}, {int64(2), nil}, {int64(3), nil}}, }, { WriteQuery: "UPDATE mytable SET s = 'updated' ORDER BY i ASC LIMIT 2;", ExpectedWriteResult: []sql.Row{{newUpdateResult(2, 2)}}, SelectQuery: "SELECT * FROM mytable;", ExpectedSelect: []sql.Row{{int64(1), "updated"}, {int64(2), "updated"}, {int64(3), "third row"}}, }, { WriteQuery: "UPDATE mytable SET s = 'updated' ORDER BY i DESC LIMIT 2;", ExpectedWriteResult: []sql.Row{{newUpdateResult(2, 2)}}, SelectQuery: "SELECT * FROM mytable;", ExpectedSelect: []sql.Row{{int64(1), "first row"}, {int64(2), "updated"}, {int64(3), "updated"}}, }, { WriteQuery: "UPDATE mytable SET s = 'updated' ORDER BY i LIMIT 1 OFFSET 1;", ExpectedWriteResult: []sql.Row{{newUpdateResult(1, 1)}}, SelectQuery: "SELECT * FROM mytable;", ExpectedSelect: []sql.Row{{int64(1), "first row"}, {int64(2), "updated"}, {int64(3), "third row"}}, }, { WriteQuery: "UPDATE mytable SET s = 'updated';", ExpectedWriteResult: []sql.Row{{newUpdateResult(3, 3)}}, SelectQuery: "SELECT * FROM mytable;", ExpectedSelect: []sql.Row{{int64(1), "updated"}, {int64(2), "updated"}, {int64(3), "updated"}}, }, { WriteQuery: "UPDATE mytable SET s = _binary 'updated' WHERE i = 3;", ExpectedWriteResult: []sql.Row{{newUpdateResult(1, 1)}}, SelectQuery: "SELECT * FROM mytable;", ExpectedSelect: []sql.Row{{int64(1), "first row"}, {int64(2), "second row"}, {int64(3), "updated"}}, }, { WriteQuery: "UPDATE typestable SET ti = '2020-03-06 00:00:00';", ExpectedWriteResult: []sql.Row{{newUpdateResult(1, 1)}}, SelectQuery: "SELECT * FROM typestable;", ExpectedSelect: []sql.Row{{ int64(1), int8(2), int16(3), int32(4), int64(5), uint8(6), uint16(7), uint32(8), uint64(9), float32(10), float64(11), sql.MustConvert(sql.Timestamp.Convert("2020-03-06 00:00:00")), sql.MustConvert(sql.Date.Convert("2019-12-31")), "fourteen", 0, nil, nil}}, }, { WriteQuery: "UPDATE typestable SET ti = '2020-03-06 00:00:00', da = '2020-03-06';", ExpectedWriteResult: []sql.Row{{newUpdateResult(1, 1)}}, SelectQuery: "SELECT * FROM typestable;", ExpectedSelect: []sql.Row{{ int64(1), int8(2), int16(3), int32(4), int64(5), uint8(6), uint16(7), uint32(8), uint64(9), float32(10), float64(11), sql.MustConvert(sql.Timestamp.Convert("2020-03-06 00:00:00")), sql.MustConvert(sql.Date.Convert("2020-03-06")), "fourteen", 0, nil, nil}}, }, { WriteQuery: "UPDATE typestable SET da = '0000-00-00', ti = '0000-00-00 00:00:00';", ExpectedWriteResult: []sql.Row{{newUpdateResult(1, 1)}}, SelectQuery: "SELECT * FROM typestable;", ExpectedSelect: []sql.Row{{ int64(1), int8(2), int16(3), int32(4), int64(5), uint8(6), uint16(7), uint32(8), uint64(9), float32(10), float64(11), sql.Timestamp.Zero(), sql.Date.Zero(), "fourteen", 0, nil, nil}}, }, { WriteQuery: `UPDATE one_pk INNER JOIN two_pk on one_pk.pk = two_pk.pk1 SET two_pk.c1 = two_pk.c1 + 1`, ExpectedWriteResult: []sql.Row{{newUpdateResult(4, 4)}}, SelectQuery: "SELECT * FROM two_pk;", ExpectedSelect: []sql.Row{ sql.NewRow(0, 0, 1, 1, 2, 3, 4), sql.NewRow(0, 1, 11, 11, 12, 13, 14), sql.NewRow(1, 0, 21, 21, 22, 23, 24), sql.NewRow(1, 1, 31, 31, 32, 33, 34), }, }, { WriteQuery: "UPDATE mytable INNER JOIN one_pk ON mytable.i = one_pk.c5 SET mytable.i = mytable.i * 10", ExpectedWriteResult: []sql.Row{{newUpdateResult(0, 0)}}, SelectQuery: "SELECT * FROM mytable", ExpectedSelect: []sql.Row{ sql.NewRow(int64(1), "first row"), sql.NewRow(int64(2), "second row"), sql.NewRow(int64(3), "third row"), }, }, { WriteQuery: `UPDATE one_pk INNER JOIN two_pk on one_pk.pk = two_pk.pk1 SET two_pk.c1 = two_pk.c1 + 1 WHERE one_pk.c5 < 10`, ExpectedWriteResult: []sql.Row{{newUpdateResult(2, 2)}}, SelectQuery: "SELECT * FROM two_pk;", ExpectedSelect: []sql.Row{ sql.NewRow(0, 0, 1, 1, 2, 3, 4), sql.NewRow(0, 1, 11, 11, 12, 13, 14), sql.NewRow(1, 0, 20, 21, 22, 23, 24), sql.NewRow(1, 1, 30, 31, 32, 33, 34), }, }, { WriteQuery: `UPDATE one_pk INNER JOIN two_pk on one_pk.pk = two_pk.pk1 INNER JOIN othertable on othertable.i2 = two_pk.pk2 SET one_pk.c1 = one_pk.c1 + 1`, ExpectedWriteResult: []sql.Row{{newUpdateResult(2, 2)}}, SelectQuery: "SELECT * FROM one_pk;", ExpectedSelect: []sql.Row{ sql.NewRow(0, 1, 1, 2, 3, 4), sql.NewRow(1, 11, 11, 12, 13, 14), sql.NewRow(2, 20, 21, 22, 23, 24), sql.NewRow(3, 30, 31, 32, 33, 34), }, }, { WriteQuery: `UPDATE one_pk INNER JOIN (SELECT * FROM two_pk order by pk1, pk2) as t2 on one_pk.pk = t2.pk1 SET one_pk.c1 = t2.c1 + 1 where one_pk.pk < 1`, ExpectedWriteResult: []sql.Row{{newUpdateResult(1, 1)}}, SelectQuery: "SELECT * FROM one_pk where pk < 1", ExpectedSelect: []sql.Row{ sql.NewRow(0, 1, 1, 2, 3, 4), }, }, { WriteQuery: `UPDATE one_pk INNER JOIN two_pk on one_pk.pk = two_pk.pk1 SET one_pk.c1 = one_pk.c1 + 1`, ExpectedWriteResult: []sql.Row{{newUpdateResult(2, 2)}}, SelectQuery: "SELECT * FROM one_pk;", ExpectedSelect: []sql.Row{ sql.NewRow(0, 1, 1, 2, 3, 4), sql.NewRow(1, 11, 11, 12, 13, 14), sql.NewRow(2, 20, 21, 22, 23, 24), sql.NewRow(3, 30, 31, 32, 33, 34), }, }, { WriteQuery: `UPDATE one_pk INNER JOIN two_pk on one_pk.pk = two_pk.pk1 SET one_pk.c1 = one_pk.c1 + 1, one_pk.c2 = one_pk.c2 + 1 ORDER BY one_pk.pk`, ExpectedWriteResult: []sql.Row{{newUpdateResult(2, 2)}}, SelectQuery: "SELECT * FROM one_pk;", ExpectedSelect: []sql.Row{ sql.NewRow(0, 1, 2, 2, 3, 4), sql.NewRow(1, 11, 12, 12, 13, 14), sql.NewRow(2, 20, 21, 22, 23, 24), sql.NewRow(3, 30, 31, 32, 33, 34), }, }, { WriteQuery: `UPDATE one_pk INNER JOIN two_pk on one_pk.pk = two_pk.pk1 SET one_pk.c1 = one_pk.c1 + 1, two_pk.c1 = two_pk.c2 + 1`, ExpectedWriteResult: []sql.Row{{newUpdateResult(8, 6)}}, SelectQuery: "SELECT * FROM two_pk;", ExpectedSelect: []sql.Row{ sql.NewRow(0, 0, 2, 1, 2, 3, 4), sql.NewRow(0, 1, 12, 11, 12, 13, 14), sql.NewRow(1, 0, 22, 21, 22, 23, 24), sql.NewRow(1, 1, 32, 31, 32, 33, 34), }, }, { WriteQuery: `UPDATE othertable CROSS JOIN tabletest set othertable.i2 = othertable.i2 * 10`, ExpectedWriteResult: []sql.Row{{newUpdateResult(3, 3)}}, SelectQuery: "SELECT * FROM othertable order by i2", ExpectedSelect: []sql.Row{ sql.NewRow("third", 10), sql.NewRow("second", 20), sql.NewRow("first", 30), }, }, { WriteQuery: `UPDATE tabletest cross join tabletest as t2 set tabletest.i = tabletest.i * 10`, ExpectedWriteResult: []sql.Row{{newUpdateResult(3, 3)}}, SelectQuery: "SELECT * FROM tabletest order by i", ExpectedSelect: []sql.Row{ sql.NewRow(10, "first row"), sql.NewRow(20, "second row"), sql.NewRow(30, "third row"), }, }, { WriteQuery: `UPDATE othertable cross join tabletest set tabletest.i = tabletest.i * 10`, ExpectedWriteResult: []sql.Row{{newUpdateResult(3, 3)}}, SelectQuery: "SELECT * FROM tabletest order by i", ExpectedSelect: []sql.Row{ sql.NewRow(10, "first row"), sql.NewRow(20, "second row"), sql.NewRow(30, "third row"), }, }, { WriteQuery: `UPDATE one_pk INNER JOIN two_pk on one_pk.pk = two_pk.pk1 INNER JOIN two_pk a1 on one_pk.pk = two_pk.pk2 SET two_pk.c1 = two_pk.c1 + 1`, ExpectedWriteResult: []sql.Row{{newUpdateResult(2, 2)}}, SelectQuery: "SELECT * FROM two_pk order by pk1 ASC, pk2 ASC;", ExpectedSelect: []sql.Row{ sql.NewRow(0, 0, 1, 1, 2, 3, 4), sql.NewRow(0, 1, 10, 11, 12, 13, 14), sql.NewRow(1, 0, 20, 21, 22, 23, 24), sql.NewRow(1, 1, 31, 31, 32, 33, 34), }, }, { WriteQuery: `UPDATE othertable INNER JOIN tabletest on othertable.i2=3 and tabletest.i=3 SET othertable.s2 = 'fourth'`, ExpectedWriteResult: []sql.Row{{newUpdateResult(1, 1)}}, SelectQuery: "SELECT * FROM othertable order by i2", ExpectedSelect: []sql.Row{ sql.NewRow("third", 1), sql.NewRow("second", 2), sql.NewRow("fourth", 3), }, }, { WriteQuery: `UPDATE tabletest cross join tabletest as t2 set t2.i = t2.i * 10`, ExpectedWriteResult: []sql.Row{{newUpdateResult(3, 3)}}, SelectQuery: "SELECT * FROM tabletest order by i", ExpectedSelect: []sql.Row{ sql.NewRow(10, "first row"), sql.NewRow(20, "second row"), sql.NewRow(30, "third row"), }, }, { WriteQuery: `UPDATE othertable LEFT JOIN tabletest on othertable.i2=3 and tabletest.i=3 SET othertable.s2 = 'fourth'`, ExpectedWriteResult: []sql.Row{{newUpdateResult(3, 3)}}, SelectQuery: "SELECT * FROM othertable order by i2", ExpectedSelect: []sql.Row{ sql.NewRow("fourth", 1), sql.NewRow("fourth", 2), sql.NewRow("fourth", 3), }, }, { WriteQuery: `UPDATE othertable LEFT JOIN tabletest on othertable.i2=3 and tabletest.i=3 SET tabletest.s = 'fourth row', tabletest.i = tabletest.i + 1`, ExpectedWriteResult: []sql.Row{{newUpdateResult(1, 1)}}, SelectQuery: "SELECT * FROM tabletest order by i", ExpectedSelect: []sql.Row{ sql.NewRow(1, "first row"), sql.NewRow(2, "second row"), sql.NewRow(4, "fourth row"), }, }, { WriteQuery: `UPDATE othertable LEFT JOIN tabletest t3 on othertable.i2=3 and t3.i=3 SET t3.s = 'fourth row', t3.i = t3.i + 1`, ExpectedWriteResult: []sql.Row{{newUpdateResult(1, 1)}}, SelectQuery: "SELECT * FROM tabletest order by i", ExpectedSelect: []sql.Row{ sql.NewRow(1, "first row"), sql.NewRow(2, "second row"), sql.NewRow(4, "fourth row"), }, }, { WriteQuery: `UPDATE othertable LEFT JOIN tabletest on othertable.i2=3 and tabletest.i=3 LEFT JOIN one_pk on othertable.i2 = one_pk.pk SET one_pk.c1 = one_pk.c1 + 1`, ExpectedWriteResult: []sql.Row{{newUpdateResult(3, 3)}}, SelectQuery: "SELECT * FROM one_pk order by pk", ExpectedSelect: []sql.Row{ sql.NewRow(0, 0, 1, 2, 3, 4), sql.NewRow(1, 11, 11, 12, 13, 14), sql.NewRow(2, 21, 21, 22, 23, 24), sql.NewRow(3, 31, 31, 32, 33, 34), }, }, { WriteQuery: `UPDATE othertable LEFT JOIN tabletest on othertable.i2=3 and tabletest.i=3 LEFT JOIN one_pk on othertable.i2 = one_pk.pk SET one_pk.c1 = one_pk.c1 + 1 where one_pk.pk > 4`, ExpectedWriteResult: []sql.Row{{newUpdateResult(0, 0)}}, SelectQuery: "SELECT * FROM one_pk order by pk", ExpectedSelect: []sql.Row{ sql.NewRow(0, 0, 1, 2, 3, 4), sql.NewRow(1, 10, 11, 12, 13, 14), sql.NewRow(2, 20, 21, 22, 23, 24), sql.NewRow(3, 30, 31, 32, 33, 34), }, }, { WriteQuery: `UPDATE othertable LEFT JOIN tabletest on othertable.i2=3 and tabletest.i=3 LEFT JOIN one_pk on othertable.i2 = 1 and one_pk.pk = 1 SET one_pk.c1 = one_pk.c1 + 1`, ExpectedWriteResult: []sql.Row{{newUpdateResult(1, 1)}}, SelectQuery: "SELECT * FROM one_pk order by pk", ExpectedSelect: []sql.Row{ sql.NewRow(0, 0, 1, 2, 3, 4), sql.NewRow(1, 11, 11, 12, 13, 14), sql.NewRow(2, 20, 21, 22, 23, 24), sql.NewRow(3, 30, 31, 32, 33, 34), }, }, { WriteQuery: `UPDATE othertable RIGHT JOIN tabletest on othertable.i2=3 and tabletest.i=3 SET othertable.s2 = 'fourth'`, ExpectedWriteResult: []sql.Row{{newUpdateResult(1, 1)}}, SelectQuery: "SELECT * FROM othertable order by i2", ExpectedSelect: []sql.Row{ sql.NewRow("third", 1), sql.NewRow("second", 2), sql.NewRow("fourth", 3), }, }, { WriteQuery: `UPDATE othertable RIGHT JOIN tabletest on othertable.i2=3 and tabletest.i=3 SET othertable.i2 = othertable.i2 + 1`, ExpectedWriteResult: []sql.Row{{newUpdateResult(1, 1)}}, SelectQuery: "SELECT * FROM othertable order by i2", ExpectedSelect: []sql.Row{ sql.NewRow("third", 1), sql.NewRow("second", 2), sql.NewRow("first", 4), }, }, { WriteQuery: `UPDATE othertable LEFT JOIN tabletest on othertable.i2=tabletest.i RIGHT JOIN one_pk on othertable.i2 = 1 and one_pk.pk = 1 SET tabletest.s = 'updated';`, ExpectedWriteResult: []sql.Row{{newUpdateResult(1, 1)}}, SelectQuery: "SELECT * FROM tabletest order by i", ExpectedSelect: []sql.Row{ sql.NewRow(1, "updated"), sql.NewRow(2, "second row"), sql.NewRow(3, "third row"), }, }, }
var UserPrivTests = []UserPrivilegeTest{ { Name: "Basic database and table name visibility", SetUpScript: []string{ "CREATE TABLE mydb.test (pk BIGINT PRIMARY KEY);", "INSERT INTO mydb.test VALUES (1);", "CREATE USER tester@localhost;", "CREATE ROLE test_role;", "GRANT SELECT ON mydb.* TO test_role;", }, Assertions: []UserPrivilegeTestAssertion{ { User: "tester", Host: "localhost", Query: "SELECT * FROM mydb.test;/*1*/", ExpectedErr: sql.ErrDatabaseAccessDeniedForUser, }, { User: "tester", Host: "localhost", Query: "SELECT * FROM mydb.test2;/*1*/", ExpectedErr: sql.ErrDatabaseAccessDeniedForUser, }, { User: "root", Host: "localhost", Query: "GRANT SELECT ON *.* TO tester@localhost;", Expected: []sql.Row{{sql.NewOkResult(0)}}, }, { User: "tester", Host: "localhost", Query: "SELECT * FROM mydb.test;/*2*/", Expected: []sql.Row{{1}}, }, { User: "tester", Host: "localhost", Query: "SELECT * FROM mydb.test2;/*2*/", ExpectedErr: sql.ErrTableNotFound, }, { User: "root", Host: "localhost", Query: "REVOKE SELECT ON *.* FROM tester@localhost;", Expected: []sql.Row{{sql.NewOkResult(0)}}, }, { User: "tester", Host: "localhost", Query: "SELECT * FROM mydb.test;/*3*/", ExpectedErr: sql.ErrDatabaseAccessDeniedForUser, }, { User: "tester", Host: "localhost", Query: "SELECT * FROM mydb.test2;/*3*/", ExpectedErr: sql.ErrDatabaseAccessDeniedForUser, }, { User: "root", Host: "localhost", Query: "GRANT SELECT ON mydb.* TO tester@localhost;", Expected: []sql.Row{{sql.NewOkResult(0)}}, }, { User: "tester", Host: "localhost", Query: "SELECT * FROM mydb.test;/*4*/", Expected: []sql.Row{{1}}, }, { User: "tester", Host: "localhost", Query: "SELECT * FROM mydb.test2;/*4*/", ExpectedErr: sql.ErrTableNotFound, }, { User: "root", Host: "localhost", Query: "REVOKE SELECT ON mydb.* FROM tester@localhost;", Expected: []sql.Row{{sql.NewOkResult(0)}}, }, { User: "tester", Host: "localhost", Query: "SELECT * FROM mydb.test;/*5*/", ExpectedErr: sql.ErrDatabaseAccessDeniedForUser, }, { User: "tester", Host: "localhost", Query: "SELECT * FROM mydb.test2;/*5*/", ExpectedErr: sql.ErrDatabaseAccessDeniedForUser, }, { User: "root", Host: "localhost", Query: "GRANT SELECT ON mydb.test TO tester@localhost;", Expected: []sql.Row{{sql.NewOkResult(0)}}, }, { User: "tester", Host: "localhost", Query: "SELECT * FROM mydb.test;/*6*/", Expected: []sql.Row{{1}}, }, { User: "tester", Host: "localhost", Query: "SELECT * FROM mydb.test2;/*6*/", ExpectedErr: sql.ErrTableAccessDeniedForUser, }, { User: "root", Host: "localhost", Query: "REVOKE SELECT ON mydb.test FROM tester@localhost;", Expected: []sql.Row{{sql.NewOkResult(0)}}, }, { User: "tester", Host: "localhost", Query: "SELECT * FROM mydb.test;/*7*/", ExpectedErr: sql.ErrDatabaseAccessDeniedForUser, }, { User: "tester", Host: "localhost", Query: "SELECT * FROM mydb.test2;/*7*/", ExpectedErr: sql.ErrDatabaseAccessDeniedForUser, }, { User: "root", Host: "localhost", Query: "GRANT SELECT ON mydb.test2 TO tester@localhost;", Expected: []sql.Row{{sql.NewOkResult(0)}}, }, { User: "tester", Host: "localhost", Query: "SELECT * FROM mydb.test;/*8*/", ExpectedErr: sql.ErrTableAccessDeniedForUser, }, { User: "tester", Host: "localhost", Query: "SELECT * FROM mydb.test2;/*8*/", ExpectedErr: sql.ErrTableNotFound, }, { User: "root", Host: "localhost", Query: "REVOKE SELECT ON mydb.test2 FROM tester@localhost;", Expected: []sql.Row{{sql.NewOkResult(0)}}, }, { User: "tester", Host: "localhost", Query: "SELECT * FROM mydb.test;/*9*/", ExpectedErr: sql.ErrDatabaseAccessDeniedForUser, }, { User: "tester", Host: "localhost", Query: "SELECT * FROM mydb.test2;/*9*/", ExpectedErr: sql.ErrDatabaseAccessDeniedForUser, }, { User: "root", Host: "localhost", Query: "GRANT test_role TO tester@localhost;", Expected: []sql.Row{{sql.NewOkResult(0)}}, }, { User: "tester", Host: "localhost", Query: "SELECT * FROM mydb.test;/*10*/", Expected: []sql.Row{{1}}, }, { User: "tester", Host: "localhost", Query: "SELECT * FROM mydb.test2;/*10*/", ExpectedErr: sql.ErrTableNotFound, }, }, }, { Name: "Basic user creation", SetUpScript: []string{ "CREATE USER testuser@`127.0.0.1`;", }, Assertions: []UserPrivilegeTestAssertion{ { Query: "CREATE USER testuser@`127.0.0.1`;", ExpectedErr: sql.ErrUserCreationFailure, }, { Query: "CREATE USER IF NOT EXISTS testuser@`127.0.0.1`;", Expected: []sql.Row{{sql.NewOkResult(0)}}, }, { Query: "INSERT INTO mysql.user (Host, User) VALUES ('localhost', 'testuser2');", Expected: []sql.Row{{sql.NewOkResult(1)}}, }, { Query: "SELECT * FROM mysql.user WHERE User = 'root';", Expected: []sql.Row{ { "localhost", "root", uint16(2), uint16(2), uint16(2), uint16(2), uint16(2), uint16(2), uint16(2), uint16(2), uint16(2), uint16(2), uint16(2), uint16(2), uint16(2), uint16(2), uint16(2), uint16(2), uint16(2), uint16(2), uint16(2), uint16(2), uint16(2), uint16(2), uint16(2), uint16(2), uint16(2), uint16(2), uint16(2), uint16(2), uint16(2), uint16(1), "", "", "", uint32(0), uint32(0), uint32(0), uint32(0), "mysql_native_password", "", uint16(1), time.Unix(1, 0).UTC(), nil, uint16(1), uint16(2), uint16(2), nil, nil, nil, nil, }, }, }, { Query: "SELECT Host, User FROM mysql.user;", Expected: []sql.Row{ {"localhost", "root"}, {"localhost", "testuser2"}, {"127.0.0.1", "testuser"}, }, }, }, }, { Name: "Valid users without privileges may use the dual table", SetUpScript: []string{ "CREATE USER tester@localhost;", }, Assertions: []UserPrivilegeTestAssertion{ { User: "tester", Host: "localhost", Query: "SELECT 1+2;", Expected: []sql.Row{{3}}, }, { User: "noexist", Host: "localhost", Query: "SELECT 1+2;", ExpectedErrStr: "Access denied for user 'noexist' (errno 1045) (sqlstate 28000)", }, }, }, { Name: "Basic SELECT and INSERT privilege checking", SetUpScript: []string{ "CREATE TABLE test (pk BIGINT PRIMARY KEY);", "INSERT INTO test VALUES (1), (2), (3);", "CREATE USER tester@localhost;", }, Assertions: []UserPrivilegeTestAssertion{ { User: "tester", Host: "localhost", Query: "INSERT INTO test VALUES (4);", ExpectedErr: sql.ErrDatabaseAccessDeniedForUser, }, { User: "root", Host: "localhost", Query: "GRANT INSERT ON *.* TO tester@localhost;", Expected: []sql.Row{{sql.NewOkResult(0)}}, }, { User: "tester", Host: "localhost", Query: "INSERT INTO test VALUES (4);", Expected: []sql.Row{{sql.NewOkResult(1)}}, }, { User: "tester", Host: "localhost", Query: "SELECT * FROM test;", ExpectedErr: sql.ErrPrivilegeCheckFailed, }, { User: "root", Host: "localhost", Query: "SELECT * FROM test;", Expected: []sql.Row{{1}, {2}, {3}, {4}}, }, { User: "root", Host: "localhost", Query: "GRANT SELECT ON *.* TO tester@localhost;", Expected: []sql.Row{{sql.NewOkResult(0)}}, }, { User: "tester", Host: "localhost", Query: "SELECT * FROM test;", Expected: []sql.Row{{1}, {2}, {3}, {4}}, }, }, }, { Name: "Database-level privileges exist", SetUpScript: []string{ "CREATE USER tester@localhost;", }, Assertions: []UserPrivilegeTestAssertion{ { User: "root", Host: "localhost", Query: "GRANT SELECT, UPDATE, EXECUTE ON mydb.* TO tester@localhost;", Expected: []sql.Row{{sql.NewOkResult(0)}}, }, { User: "root", Host: "localhost", Query: "SELECT * FROM mysql.db;", Expected: []sql.Row{{"localhost", "mydb", "tester", uint16(2), uint16(1), uint16(2), uint16(1), uint16(1), uint16(1), uint16(1), uint16(1), uint16(1), uint16(1), uint16(1), uint16(1), uint16(1), uint16(1), uint16(1), uint16(1), uint16(2), uint16(1), uint16(1)}}, }, { User: "root", Host: "localhost", Query: "REVOKE UPDATE ON mydb.* FROM tester@localhost;", Expected: []sql.Row{{sql.NewOkResult(0)}}, }, { User: "root", Host: "localhost", Query: "SELECT * FROM mysql.db;", Expected: []sql.Row{{"localhost", "mydb", "tester", uint16(2), uint16(1), uint16(1), uint16(1), uint16(1), uint16(1), uint16(1), uint16(1), uint16(1), uint16(1), uint16(1), uint16(1), uint16(1), uint16(1), uint16(1), uint16(1), uint16(2), uint16(1), uint16(1)}}, }, { User: "root", Host: "localhost", Query: "UPDATE mysql.db SET Insert_priv = 'Y' WHERE User = 'tester';", Expected: []sql.Row{{sql.OkResult{ RowsAffected: 1, InsertID: 0, Info: plan.UpdateInfo{ Matched: 1, Updated: 1, Warnings: 0, }, }}}, }, { User: "root", Host: "localhost", Query: "SELECT * FROM mysql.db;", Expected: []sql.Row{{"localhost", "mydb", "tester", uint16(2), uint16(2), uint16(1), uint16(1), uint16(1), uint16(1), uint16(1), uint16(1), uint16(1), uint16(1), uint16(1), uint16(1), uint16(1), uint16(1), uint16(1), uint16(1), uint16(2), uint16(1), uint16(1)}}, }, }, }, { Name: "Table-level privileges exist", SetUpScript: []string{ "CREATE TABLE test (pk BIGINT PRIMARY KEY);", "CREATE USER tester@localhost;", }, Assertions: []UserPrivilegeTestAssertion{ { User: "root", Host: "localhost", Query: "GRANT SELECT, DELETE, DROP ON mydb.test TO tester@localhost;", Expected: []sql.Row{{sql.NewOkResult(0)}}, }, { User: "root", Host: "localhost", Query: "SELECT * FROM mysql.tables_priv;", Expected: []sql.Row{{"localhost", "mydb", "tester", "test", "", time.Unix(1, 0).UTC(), uint64(0b101001), uint64(0)}}, }, { User: "root", Host: "localhost", Query: "REVOKE DELETE ON mydb.test FROM tester@localhost;", Expected: []sql.Row{{sql.NewOkResult(0)}}, }, { User: "root", Host: "localhost", Query: "SELECT * FROM mysql.tables_priv;", Expected: []sql.Row{{"localhost", "mydb", "tester", "test", "", time.Unix(1, 0).UTC(), uint64(0b100001), uint64(0)}}, }, { User: "root", Host: "localhost", Query: "UPDATE mysql.tables_priv SET table_priv = 'References,Index' WHERE User = 'tester';", Expected: []sql.Row{{sql.OkResult{ RowsAffected: 1, InsertID: 0, Info: plan.UpdateInfo{ Matched: 1, Updated: 1, Warnings: 0, }, }}}, }, { User: "root", Host: "localhost", Query: "SELECT * FROM mysql.tables_priv;", Expected: []sql.Row{{"localhost", "mydb", "tester", "test", "", time.Unix(1, 0).UTC(), uint64(0b110000000), uint64(0)}}, }, }, }, { Name: "Basic revoke SELECT privilege", SetUpScript: []string{ "CREATE TABLE test (pk BIGINT PRIMARY KEY);", "INSERT INTO test VALUES (1), (2), (3);", "CREATE USER tester@localhost;", "GRANT SELECT ON *.* TO tester@localhost;", }, Assertions: []UserPrivilegeTestAssertion{ { User: "tester", Host: "localhost", Query: "SELECT * FROM test;", Expected: []sql.Row{{1}, {2}, {3}}, }, { User: "root", Host: "localhost", Query: "SELECT User, Host, Select_priv FROM mysql.user WHERE User = 'tester';", Expected: []sql.Row{{"tester", "localhost", uint16(2)}}, }, { User: "root", Host: "localhost", Query: "REVOKE SELECT ON *.* FROM tester@localhost;", Expected: []sql.Row{{sql.NewOkResult(0)}}, }, { User: "tester", Host: "localhost", Query: "SELECT * FROM test;", ExpectedErr: sql.ErrDatabaseAccessDeniedForUser, }, { User: "root", Host: "localhost", Query: "SELECT User, Host, Select_priv FROM mysql.user WHERE User = 'tester';", Expected: []sql.Row{{"tester", "localhost", uint16(1)}}, }, }, }, { Name: "Basic revoke all global static privileges", SetUpScript: []string{ "CREATE TABLE test (pk BIGINT PRIMARY KEY);", "INSERT INTO test VALUES (1), (2), (3);", "CREATE USER tester@localhost;", "GRANT ALL ON *.* TO tester@localhost;", }, Assertions: []UserPrivilegeTestAssertion{ { User: "tester", Host: "localhost", Query: "INSERT INTO test VALUES (4);", Expected: []sql.Row{{sql.NewOkResult(1)}}, }, { User: "tester", Host: "localhost", Query: "SELECT * FROM test;", Expected: []sql.Row{{1}, {2}, {3}, {4}}, }, { User: "root", Host: "localhost", Query: "SELECT User, Host, Select_priv, Insert_priv FROM mysql.user WHERE User = 'tester';", Expected: []sql.Row{{"tester", "localhost", uint16(2), uint16(2)}}, }, { User: "root", Host: "localhost", Query: "REVOKE ALL ON *.* FROM tester@localhost;", Expected: []sql.Row{{sql.NewOkResult(0)}}, }, { User: "tester", Host: "localhost", Query: "SELECT * FROM test;", ExpectedErr: sql.ErrDatabaseAccessDeniedForUser, }, { User: "tester", Host: "localhost", Query: "INSERT INTO test VALUES (5);", ExpectedErr: sql.ErrDatabaseAccessDeniedForUser, }, { User: "root", Host: "localhost", Query: "SELECT User, Host, Select_priv, Insert_priv FROM mysql.user WHERE User = 'tester';", Expected: []sql.Row{{"tester", "localhost", uint16(1), uint16(1)}}, }, }, }, { Name: "Basic role creation", SetUpScript: []string{ "CREATE ROLE test_role;", }, Assertions: []UserPrivilegeTestAssertion{ { User: "root", Host: "localhost", Query: "SELECT User, Host, account_locked FROM mysql.user WHERE User = 'test_role';", Expected: []sql.Row{{"test_role", "%", uint16(2)}}, }, }, }, { Name: "Grant Role with SELECT Privilege", SetUpScript: []string{ "SET @@GLOBAL.activate_all_roles_on_login = true;", "CREATE TABLE test (pk BIGINT PRIMARY KEY);", "INSERT INTO test VALUES (1), (2), (3);", "CREATE USER tester@localhost;", "CREATE ROLE test_role;", "GRANT SELECT ON *.* TO test_role;", }, Assertions: []UserPrivilegeTestAssertion{ { User: "tester", Host: "localhost", Query: "SELECT * FROM test;", ExpectedErr: sql.ErrDatabaseAccessDeniedForUser, }, { User: "root", Host: "localhost", Query: "SELECT COUNT(*) FROM mysql.role_edges;", Expected: []sql.Row{{0}}, }, { User: "root", Host: "localhost", Query: "GRANT test_role TO tester@localhost;", Expected: []sql.Row{{sql.NewOkResult(0)}}, }, { User: "root", Host: "localhost", Query: "SELECT * FROM mysql.role_edges;", Expected: []sql.Row{{"%", "test_role", "localhost", "tester", uint16(1)}}, }, { User: "tester", Host: "localhost", Query: "SELECT * FROM test;", Expected: []sql.Row{{1}, {2}, {3}}, }, { User: "root", Host: "localhost", Query: "SELECT User, Host, Select_priv FROM mysql.user WHERE User = 'tester';", Expected: []sql.Row{{"tester", "localhost", uint16(1)}}, }, }, }, { Name: "Revoke role currently granted to a user", SetUpScript: []string{ "SET @@GLOBAL.activate_all_roles_on_login = true;", "CREATE TABLE test (pk BIGINT PRIMARY KEY);", "INSERT INTO test VALUES (1), (2), (3);", "CREATE USER tester@localhost;", "CREATE ROLE test_role;", "GRANT SELECT ON *.* TO test_role;", "GRANT test_role TO tester@localhost;", }, Assertions: []UserPrivilegeTestAssertion{ { User: "tester", Host: "localhost", Query: "SELECT * FROM test;", Expected: []sql.Row{{1}, {2}, {3}}, }, { User: "root", Host: "localhost", Query: "SELECT * FROM mysql.role_edges;", Expected: []sql.Row{{"%", "test_role", "localhost", "tester", uint16(1)}}, }, { User: "root", Host: "localhost", Query: "REVOKE test_role FROM tester@localhost;", Expected: []sql.Row{{sql.NewOkResult(0)}}, }, { User: "tester", Host: "localhost", Query: "SELECT * FROM test;", ExpectedErr: sql.ErrDatabaseAccessDeniedForUser, }, { User: "root", Host: "localhost", Query: "SELECT COUNT(*) FROM mysql.role_edges;", Expected: []sql.Row{{0}}, }, { User: "root", Host: "localhost", Query: "SELECT COUNT(*) FROM mysql.user WHERE User = 'test_role';", Expected: []sql.Row{{1}}, }, { User: "root", Host: "localhost", Query: "SELECT COUNT(*) FROM mysql.user WHERE User = 'tester';", Expected: []sql.Row{{1}}, }, }, }, { Name: "Drop role currently granted to a user", SetUpScript: []string{ "SET @@GLOBAL.activate_all_roles_on_login = true;", "CREATE TABLE test (pk BIGINT PRIMARY KEY);", "INSERT INTO test VALUES (1), (2), (3);", "CREATE USER tester@localhost;", "CREATE ROLE test_role;", "GRANT SELECT ON *.* TO test_role;", "GRANT test_role TO tester@localhost;", }, Assertions: []UserPrivilegeTestAssertion{ { User: "tester", Host: "localhost", Query: "SELECT * FROM test;", Expected: []sql.Row{{1}, {2}, {3}}, }, { User: "root", Host: "localhost", Query: "SELECT * FROM mysql.role_edges;", Expected: []sql.Row{{"%", "test_role", "localhost", "tester", uint16(1)}}, }, { User: "root", Host: "localhost", Query: "DROP ROLE test_role;", Expected: []sql.Row{{sql.NewOkResult(0)}}, }, { User: "tester", Host: "localhost", Query: "SELECT * FROM test;", ExpectedErr: sql.ErrDatabaseAccessDeniedForUser, }, { User: "root", Host: "localhost", Query: "SELECT COUNT(*) FROM mysql.role_edges;", Expected: []sql.Row{{0}}, }, { User: "root", Host: "localhost", Query: "SELECT COUNT(*) FROM mysql.user WHERE User = 'test_role';", Expected: []sql.Row{{0}}, }, { User: "root", Host: "localhost", Query: "SELECT COUNT(*) FROM mysql.user WHERE User = 'tester';", Expected: []sql.Row{{1}}, }, { User: "root", Host: "localhost", Query: "DROP ROLE test_role;", ExpectedErr: sql.ErrRoleDeletionFailure, }, { User: "root", Host: "localhost", Query: "DROP ROLE IF EXISTS test_role;", Expected: []sql.Row{{sql.NewOkResult(0)}}, }, }, }, { Name: "Drop user with role currently granted", SetUpScript: []string{ "SET @@GLOBAL.activate_all_roles_on_login = true;", "CREATE TABLE test (pk BIGINT PRIMARY KEY);", "INSERT INTO test VALUES (1), (2), (3);", "CREATE USER tester@localhost;", "CREATE ROLE test_role;", "GRANT SELECT ON *.* TO test_role;", "GRANT test_role TO tester@localhost;", }, Assertions: []UserPrivilegeTestAssertion{ { User: "root", Host: "localhost", Query: "SELECT * FROM mysql.role_edges;", Expected: []sql.Row{{"%", "test_role", "localhost", "tester", uint16(1)}}, }, { User: "root", Host: "localhost", Query: "DROP USER tester@localhost;", Expected: []sql.Row{{sql.NewOkResult(0)}}, }, { User: "root", Host: "localhost", Query: "SELECT COUNT(*) FROM mysql.role_edges;", Expected: []sql.Row{{0}}, }, { User: "root", Host: "localhost", Query: "SELECT COUNT(*) FROM mysql.user WHERE User = 'tester';", Expected: []sql.Row{{0}}, }, { User: "root", Host: "localhost", Query: "SELECT COUNT(*) FROM mysql.user WHERE User = 'test_role';", Expected: []sql.Row{{1}}, }, { User: "root", Host: "localhost", Query: "DROP USER tester@localhost;", ExpectedErr: sql.ErrUserDeletionFailure, }, { User: "root", Host: "localhost", Query: "DROP USER IF EXISTS tester@localhost;", Expected: []sql.Row{{sql.NewOkResult(0)}}, }, }, }, { Name: "Show grants on root account", Assertions: []UserPrivilegeTestAssertion{ { User: "root", Host: "localhost", Query: "SHOW GRANTS;", Expected: []sql.Row{{"GRANT SELECT, INSERT, UPDATE, DELETE, CREATE, DROP, RELOAD, SHUTDOWN, PROCESS, " + "FILE, REFERENCES, INDEX, ALTER, SHOW DATABASES, SUPER, CREATE TEMPORARY TABLES, LOCK TABLES, " + "EXECUTE, REPLICATION SLAVE, REPLICATION CLIENT, CREATE VIEW, SHOW VIEW, CREATE ROUTINE, " + "ALTER ROUTINE, CREATE USER, EVENT, TRIGGER, CREATE TABLESPACE, CREATE ROLE, DROP ROLE ON *.* TO " + "`root`@`localhost` WITH GRANT OPTION"}}, }, }, }, { Name: "Show grants on a user from the root account", SetUpScript: []string{ "CREATE USER tester@localhost;", "GRANT SELECT ON *.* TO tester@localhost;", "CREATE ROLE test_role1;", "CREATE ROLE test_role2;", "GRANT INSERT ON *.* TO test_role1;", "GRANT REFERENCES ON *.* TO test_role2;", "GRANT test_role1 TO tester@localhost;", "GRANT test_role2 TO tester@localhost;", }, Assertions: []UserPrivilegeTestAssertion{ { User: "root", Host: "localhost", Query: "SHOW GRANTS FOR tester@localhost;", Expected: []sql.Row{ {"GRANT SELECT ON *.* TO `tester`@`localhost`"}, {"GRANT `test_role1`@`%`, `test_role2`@`%` TO `tester`@`localhost`"}, }, }, { User: "root", Host: "localhost", Query: "GRANT UPDATE ON *.* TO tester@localhost WITH GRANT OPTION;", Expected: []sql.Row{{sql.NewOkResult(0)}}, }, { User: "root", Host: "localhost", Query: "SHOW GRANTS FOR tester@localhost;", Expected: []sql.Row{ {"GRANT SELECT, UPDATE ON *.* TO `tester`@`localhost` WITH GRANT OPTION"}, {"GRANT `test_role1`@`%`, `test_role2`@`%` TO `tester`@`localhost`"}, }, }, { User: "tester", Host: "localhost", Query: "SHOW GRANTS;", Expected: []sql.Row{ {"GRANT SELECT, UPDATE ON *.* TO `tester`@`localhost` WITH GRANT OPTION"}, {"GRANT `test_role1`@`%`, `test_role2`@`%` TO `tester`@`localhost`"}, }, }, }, }, }
UserPrivTests test the user and privilege systems. These tests always have the root account available, and the root account is used with any queries in the SetUpScript.
var VariableErrorTests = []QueryErrorTest{ { Query: "set @@does_not_exist = 100", ExpectedErr: sql.ErrUnknownSystemVariable, }, { Query: "set @myvar = bareword", ExpectedErr: sql.ErrColumnNotFound, }, { Query: "set @@sql_mode = true", ExpectedErr: sql.ErrInvalidSystemVariableValue, }, { Query: `set @@sql_mode = "NOT_AN_OPTION"`, ExpectedErr: sql.ErrInvalidSetValue, }, { Query: `set global core_file = true`, ExpectedErr: sql.ErrSystemVariableReadOnly, }, { Query: `set global require_row_format = on`, ExpectedErr: sql.ErrSystemVariableSessionOnly, }, { Query: `set session default_password_lifetime = 5`, ExpectedErr: sql.ErrSystemVariableGlobalOnly, }, { Query: `set @custom_var = default`, ExpectedErr: sql.ErrUserVariableNoDefault, }, { Query: `set session @@bulk_insert_buffer_size = 5`, ExpectedErr: sql.ErrSyntaxError, }, { Query: `set global @@bulk_insert_buffer_size = 5`, ExpectedErr: sql.ErrSyntaxError, }, { Query: `set session @@session.bulk_insert_buffer_size = 5`, ExpectedErr: sql.ErrSyntaxError, }, { Query: `set session @@global.bulk_insert_buffer_size = 5`, ExpectedErr: sql.ErrSyntaxError, }, { Query: `set global @@session.bulk_insert_buffer_size = 5`, ExpectedErr: sql.ErrSyntaxError, }, { Query: `set global @@global.bulk_insert_buffer_size = 5`, ExpectedErr: sql.ErrSyntaxError, }, { Query: `set session @myvar = 5`, ExpectedErr: sql.ErrSyntaxError, }, { Query: `set global @myvar = 5`, ExpectedErr: sql.ErrSyntaxError, }, { Query: `set @@session.@@bulk_insert_buffer_size = 5`, ExpectedErr: sql.ErrSyntaxError, }, { Query: `set @@global.@@bulk_insert_buffer_size = 5`, ExpectedErr: sql.ErrSyntaxError, }, { Query: `set @@session.@bulk_insert_buffer_size = 5`, ExpectedErr: sql.ErrSyntaxError, }, { Query: `set @@global.@bulk_insert_buffer_size = 5`, ExpectedErr: sql.ErrSyntaxError, }, { Query: `set @@session.@myvar = 5`, ExpectedErr: sql.ErrSyntaxError, }, { Query: `set @@global.@myvar = 5`, ExpectedErr: sql.ErrSyntaxError, }, }
var VariableQueries = []ScriptTest{ { Name: "set system variables", SetUpScript: []string{ "set @@auto_increment_increment = 100, sql_select_limit = 1", }, Query: "SELECT @@auto_increment_increment, @@sql_select_limit", Expected: []sql.Row{ {100, 1}, }, }, { Name: "set system variables and user variables", SetUpScript: []string{ "SET @myvar = @@autocommit", "SET autocommit = @myvar", "SET @myvar2 = @myvar - 1, @myvar3 = @@autocommit - 1", }, Assertions: []ScriptTestAssertion{ { Query: "select @myvar, @@autocommit, @myvar2, @myvar3", Expected: []sql.Row{ {1, 1, 0, 0.0}, }, }, }, }, { Name: "set system variables mixed case", SetUpScript: []string{ "set @@auto_increment_INCREMENT = 100, sql_select_LIMIT = 1", }, Query: "SELECT @@auto_increment_increment, @@sql_select_limit", Expected: []sql.Row{ {100, 1}, }, }, { Name: "set system variable defaults", SetUpScript: []string{ "set @@auto_increment_increment = 100, sql_select_limit = 1", "set @@auto_increment_increment = default, sql_select_limit = default", }, Query: "SELECT @@auto_increment_increment, @@sql_select_limit", Expected: []sql.Row{ {1, math.MaxInt32}, }, }, { Name: "set system variable ON / OFF", SetUpScript: []string{ "set @@autocommit = ON, sql_mode = \"\"", }, Query: "SELECT @@autocommit, @@session.sql_mode", Expected: []sql.Row{ {1, uint64(0)}, }, }, { Name: "set system variable ON / OFF", SetUpScript: []string{ "set @@autocommit = ON, session sql_mode = \"\"", }, Query: "SELECT @@autocommit, @@session.sql_mode", Expected: []sql.Row{ {1, uint64(0)}, }, }, { Name: "set system variable true / false quoted", SetUpScript: []string{ `set @@autocommit = "true", default_table_encryption = "false"`, }, Query: "SELECT @@autocommit, @@session.default_table_encryption", Expected: []sql.Row{ {1, 0}, }, }, { Name: "set system variable true / false", SetUpScript: []string{ `set @@autocommit = true, default_table_encryption = false`, }, Query: "SELECT @@autocommit, @@session.default_table_encryption", Expected: []sql.Row{ {1, 0}, }, }, { Name: "set system variable with expressions", SetUpScript: []string{ `set lc_messages = "123", @@auto_increment_increment = 1`, `set lc_messages = concat(@@lc_messages, "456"), @@auto_increment_increment = @@auto_increment_increment + 3`, }, Query: "SELECT @@lc_messages, @@auto_increment_increment", Expected: []sql.Row{ {"123456", 4}, }, }, { Name: "set system variable to another system variable", SetUpScript: []string{ `set @@auto_increment_increment = 123`, `set @@sql_select_limit = @@auto_increment_increment`, }, Query: "SELECT @@sql_select_limit", Expected: []sql.Row{ {123}, }, }, { Name: "set names", SetUpScript: []string{ `set names utf8mb4`, }, Query: "SELECT @@character_set_client, @@character_set_connection, @@character_set_results", Expected: []sql.Row{ {"utf8mb4", "utf8mb4", "utf8mb4"}, }, }, { Name: "set names quoted", SetUpScript: []string{ `set NAMES "charset"`, }, Query: "SELECT @@character_set_client, @@character_set_connection, @@character_set_results", Expected: []sql.Row{ {"charset", "charset", "charset"}, }, }, { Name: "set character set", SetUpScript: []string{ `set character set utf8`, }, Query: "SELECT @@character_set_client, @@character_set_connection, @@character_set_results", Expected: []sql.Row{ {"utf8", "utf8mb4", "utf8"}, }, }, { Name: "set charset", SetUpScript: []string{ `set charset utf8`, }, Query: "SELECT @@character_set_client, @@character_set_connection, @@character_set_results", Expected: []sql.Row{ {"utf8", "utf8mb4", "utf8"}, }, }, { Name: "set charset quoted", SetUpScript: []string{ `set charset 'utf8'`, }, Query: "SELECT @@character_set_client, @@character_set_connection, @@character_set_results", Expected: []sql.Row{ {"utf8", "utf8mb4", "utf8"}, }, }, { Name: "set system variable to bareword", SetUpScript: []string{ `set @@sql_mode = ALLOW_INVALID_DATES`, }, Query: "SELECT @@sql_mode", Expected: []sql.Row{ {uint64(1)}, }, }, { Name: "set system variable to bareword, unqualified", SetUpScript: []string{ `set sql_mode = ALLOW_INVALID_DATES`, }, Query: "SELECT @@sql_mode", Expected: []sql.Row{ {uint64(1)}, }, }, { Name: "set sql_mode variable from mysqldump", SetUpScript: []string{ `SET sql_mode = 'STRICT_TRANS_TABLES,STRICT_ALL_TABLES,NO_ZERO_IN_DATE,NO_ZERO_DATE,ERROR_FOR_DIVISION_BY_ZERO,TRADITIONAL,NO_ENGINE_SUBSTITUTION'`, }, Query: "SELECT @@sql_mode", Expected: []sql.Row{ {uint64(0b10110000110100000100)}, }, }, { Name: "set user var", SetUpScript: []string{ `set @myvar = "hello"`, }, Query: "SELECT @myvar", Expected: []sql.Row{ {"hello"}, }, }, { Name: "set user var, integer type", SetUpScript: []string{ `set @myvar = 123`, }, Query: "SELECT @myvar", Expected: []sql.Row{ {123}, }, }, { Name: "set user var, floating point", SetUpScript: []string{ `set @myvar = 123.4`, }, Query: "SELECT @myvar", Expected: []sql.Row{ {123.4}, }, }, { Name: "set user var and sys var in same statement", SetUpScript: []string{ `set @myvar = 123.4, @@auto_increment_increment = 1234`, }, Query: "SELECT @myvar, @@auto_increment_increment", Expected: []sql.Row{ {123.4, 1234}, }, }, { Name: "set sys var to user var", SetUpScript: []string{ `set @myvar = 1234`, `set auto_increment_increment = @myvar`, }, Query: "SELECT @myvar, @@auto_increment_increment", Expected: []sql.Row{ {1234, 1234}, }, }, { Name: "local is session", SetUpScript: []string{ `set @@LOCAL.cte_max_recursion_depth = 1234`, }, Query: "SELECT @@SESSION.cte_max_recursion_depth", Expected: []sql.Row{ {1234}, }, }, { Name: "user and system var with same name", SetUpScript: []string{ `set @cte_max_recursion_depth = 55`, `set cte_max_recursion_depth = 77`, }, Query: "SELECT @cte_max_recursion_depth, @@cte_max_recursion_depth", Expected: []sql.Row{ {55, 77}, }, }, { Name: "uninitialized user vars", Assertions: []ScriptTestAssertion{ { Query: "SELECT @doesNotExist;", Expected: []sql.Row{{nil}}, }, { Query: "SELECT @doesNotExist is NULL;", Expected: []sql.Row{{true}}, }, { Query: "SELECT @doesNotExist='';", Expected: []sql.Row{{nil}}, }, { Query: "SELECT @doesNotExist < 123;", Expected: []sql.Row{{nil}}, }, }, }, { Name: "eval string user var", SetUpScript: []string{ "set @stringVar = 'abc'", }, Assertions: []ScriptTestAssertion{ { Query: "SELECT @stringVar='abc'", Expected: []sql.Row{{true}}, }, { Query: "SELECT @stringVar='abcd';", Expected: []sql.Row{{false}}, }, { Query: "SELECT @stringVar=123;", Expected: []sql.Row{{false}}, }, { Query: "SELECT @stringVar is null;", Expected: []sql.Row{{false}}, }, }, }, { Name: "set transaction", Assertions: []ScriptTestAssertion{ { Query: "set transaction isolation level serializable, read only", Expected: []sql.Row{{}}, }, { Query: "select @@transaction_isolation, @@transaction_read_only", Expected: []sql.Row{{"SERIALIZABLE", 1}}, }, { Query: "set transaction read write, isolation level read uncommitted", Expected: []sql.Row{{}}, }, { Query: "select @@transaction_isolation, @@transaction_read_only", Expected: []sql.Row{{"READ-UNCOMMITTED", 0}}, }, { Query: "set transaction isolation level read committed", Expected: []sql.Row{{}}, }, { Query: "select @@transaction_isolation", Expected: []sql.Row{{"READ-COMMITTED"}}, }, { Query: "set transaction isolation level repeatable read", Expected: []sql.Row{{}}, }, { Query: "select @@transaction_isolation", Expected: []sql.Row{{"REPEATABLE-READ"}}, }, { Query: "set session transaction isolation level serializable, read only", Expected: []sql.Row{{}}, }, { Query: "select @@transaction_isolation, @@transaction_read_only", Expected: []sql.Row{{"SERIALIZABLE", 1}}, }, { Query: "set global transaction read write, isolation level read uncommitted", Expected: []sql.Row{{}}, }, { Query: "select @@transaction_isolation, @@transaction_read_only", Expected: []sql.Row{{"SERIALIZABLE", 1}}, }, { Query: "select @@global.transaction_isolation, @@global.transaction_read_only", Expected: []sql.Row{{"READ-UNCOMMITTED", 0}}, }, }, }, }
var VersionedQueries = []QueryTest{ { Query: "SELECT * FROM myhistorytable AS OF '2019-01-01' AS foo ORDER BY i", Expected: []sql.Row{ {int64(1), "first row, 1"}, {int64(2), "second row, 1"}, {int64(3), "third row, 1"}, }, }, { Query: "SELECT * FROM myhistorytable AS OF '2019-01-02' foo ORDER BY i", Expected: []sql.Row{ {int64(1), "first row, 2"}, {int64(2), "second row, 2"}, {int64(3), "third row, 2"}, }, }, { Query: "SELECT * FROM myhistorytable AS OF GREATEST('2019-01-02','2019-01-01','') foo ORDER BY i", Expected: []sql.Row{ {int64(1), "first row, 2"}, {int64(2), "second row, 2"}, {int64(3), "third row, 2"}, }, }, { Query: "SELECT * FROM myhistorytable ORDER BY i", Expected: []sql.Row{ {int64(1), "first row, 3", "1"}, {int64(2), "second row, 3", "2"}, {int64(3), "third row, 3", "3"}, }, }, { Query: "SHOW TABLES AS OF '2019-01-02' LIKE 'myhistorytable'", Expected: []sql.Row{ {"myhistorytable"}, }, }, { Query: "SHOW TABLES FROM mydb AS OF '2019-01-02' LIKE 'myhistorytable'", Expected: []sql.Row{ {"myhistorytable"}, }, }, { Query: "SHOW CREATE TABLE myhistorytable as of '2019-01-02'", Expected: []sql.Row{ {"myhistorytable", "CREATE TABLE `myhistorytable` (\n" + " `i` bigint NOT NULL,\n" + " `s` text NOT NULL,\n" + " PRIMARY KEY (`i`)\n" + ") ENGINE=InnoDB DEFAULT CHARSET=utf8mb4 COLLATE=utf8mb4_0900_bin"}, }, }, { Query: "SHOW CREATE TABLE myhistorytable as of '2019-01-03'", Expected: []sql.Row{ {"myhistorytable", "CREATE TABLE `myhistorytable` (\n" + " `i` bigint NOT NULL,\n" + " `s` text NOT NULL,\n" + " `c` text NOT NULL,\n" + " PRIMARY KEY (`i`)\n" + ") ENGINE=InnoDB DEFAULT CHARSET=utf8mb4 COLLATE=utf8mb4_0900_bin"}, }, }, }
var VersionedScripts = []ScriptTest{ { Name: "user var for AS OF expression", SetUpScript: []string{ "SET @rev1 = '2019-01-01', @rev2 = '2019-01-02'", }, Assertions: []ScriptTestAssertion{ { Query: "SELECT * FROM myhistorytable AS OF @rev1 AS foo ORDER BY i", Expected: []sql.Row{ {int64(1), "first row, 1"}, {int64(2), "second row, 1"}, {int64(3), "third row, 1"}, }, }, { Query: "SELECT * FROM myhistorytable AS OF @rev2 AS foo ORDER BY i", Expected: []sql.Row{ {int64(1), "first row, 2"}, {int64(2), "second row, 2"}, {int64(3), "third row, 2"}, }, }, { Query: "SHOW TABLES AS OF @rev1 LIKE 'myhistorytable'", Expected: []sql.Row{ {"myhistorytable"}, }, }, { Query: "DESCRIBE myhistorytable AS OF '2019-01-02'", Expected: []sql.Row{ {"i", "bigint", "NO", "PRI", "", ""}, {"s", "text", "NO", "", "", ""}, }, }, { Query: "DESCRIBE myhistorytable AS OF '2019-01-03'", Expected: []sql.Row{ {"i", "bigint", "NO", "PRI", "", ""}, {"s", "text", "NO", "", "", ""}, {"c", "text", "NO", "", "", ""}, }, }, }, }, }
var VersionedViewTests = []QueryTest{ { Query: "SELECT * FROM myview1 ORDER BY i", Expected: []sql.Row{ sql.NewRow(int64(1), "first row, 3", "1"), sql.NewRow(int64(2), "second row, 3", "2"), sql.NewRow(int64(3), "third row, 3", "3"), }, }, { Query: "SELECT t.* FROM myview1 AS t ORDER BY i", Expected: []sql.Row{ sql.NewRow(int64(1), "first row, 3", "1"), sql.NewRow(int64(2), "second row, 3", "2"), sql.NewRow(int64(3), "third row, 3", "3"), }, }, { Query: "SELECT t.i FROM myview1 AS t ORDER BY i", Expected: []sql.Row{ sql.NewRow(int64(1)), sql.NewRow(int64(2)), sql.NewRow(int64(3)), }, }, { Query: "SELECT * FROM myview1 AS OF '2019-01-01' ORDER BY i", Expected: []sql.Row{ sql.NewRow(int64(1), "first row, 1"), sql.NewRow(int64(2), "second row, 1"), sql.NewRow(int64(3), "third row, 1"), }, }, { Query: "SELECT * FROM myview2", Expected: []sql.Row{ sql.NewRow(int64(1), "first row, 3", "1"), }, }, { Query: "SELECT i FROM myview2", Expected: []sql.Row{ sql.NewRow(int64(1)), }, }, { Query: "SELECT myview2.i FROM myview2", Expected: []sql.Row{ sql.NewRow(int64(1)), }, }, { Query: "SELECT myview2.* FROM myview2", Expected: []sql.Row{ sql.NewRow(int64(1), "first row, 3", "1"), }, }, { Query: "SELECT t.* FROM myview2 as t", Expected: []sql.Row{ sql.NewRow(int64(1), "first row, 3", "1"), }, }, { Query: "SELECT t.i FROM myview2 as t", Expected: []sql.Row{ sql.NewRow(int64(1)), }, }, { Query: "SELECT * FROM myview2 AS OF '2019-01-01'", Expected: []sql.Row{ sql.NewRow(int64(1), "first row, 1"), }, }, { Query: "select * from information_schema.views where table_schema = 'mydb'", Expected: []sql.Row{ sql.NewRow("def", "mydb", "myview", "SELECT * FROM mytable", "NONE", "YES", "", "DEFINER", "utf8mb4", "utf8mb4_0900_bin"), sql.NewRow("def", "mydb", "myview1", "SELECT * FROM myhistorytable", "NONE", "YES", "", "DEFINER", "utf8mb4", "utf8mb4_0900_bin"), sql.NewRow("def", "mydb", "myview2", "SELECT * FROM myview1 WHERE i = 1", "NONE", "YES", "", "DEFINER", "utf8mb4", "utf8mb4_0900_bin"), }, }, { Query: "select table_name from information_schema.tables where table_schema = 'mydb' and table_type = 'VIEW' order by 1", Expected: []sql.Row{ sql.NewRow("myview"), sql.NewRow("myview1"), sql.NewRow("myview2"), }, }, }
var ViewTests = []QueryTest{ { Query: "SELECT * FROM myview ORDER BY i", Expected: []sql.Row{ sql.NewRow(int64(1), "first row"), sql.NewRow(int64(2), "second row"), sql.NewRow(int64(3), "third row"), }, }, { Query: "SELECT myview.* FROM myview ORDER BY i", Expected: []sql.Row{ sql.NewRow(int64(1), "first row"), sql.NewRow(int64(2), "second row"), sql.NewRow(int64(3), "third row"), }, }, { Query: "SELECT i FROM myview ORDER BY i", Expected: []sql.Row{ sql.NewRow(int64(1)), sql.NewRow(int64(2)), sql.NewRow(int64(3)), }, }, { Query: "SELECT t.* FROM myview AS t ORDER BY i", Expected: []sql.Row{ sql.NewRow(int64(1), "first row"), sql.NewRow(int64(2), "second row"), sql.NewRow(int64(3), "third row"), }, }, { Query: "SELECT t.i FROM myview AS t ORDER BY i", Expected: []sql.Row{ sql.NewRow(int64(1)), sql.NewRow(int64(2)), sql.NewRow(int64(3)), }, }, { Query: "SELECT * FROM myview2", Expected: []sql.Row{ sql.NewRow(int64(1), "first row"), }, }, { Query: "SELECT i FROM myview2", Expected: []sql.Row{ sql.NewRow(int64(1)), }, }, { Query: "SELECT myview2.i FROM myview2", Expected: []sql.Row{ sql.NewRow(int64(1)), }, }, { Query: "SELECT myview2.* FROM myview2", Expected: []sql.Row{ sql.NewRow(int64(1), "first row"), }, }, { Query: "SELECT t.* FROM myview2 as t", Expected: []sql.Row{ sql.NewRow(int64(1), "first row"), }, }, { Query: "SELECT t.i FROM myview2 as t", Expected: []sql.Row{ sql.NewRow(int64(1)), }, }, { Query: "select * from information_schema.views where table_schema = 'mydb' order by table_name", Expected: []sql.Row{ sql.NewRow("def", "mydb", "myview", "SELECT * FROM mytable", "NONE", "YES", "", "DEFINER", "utf8mb4", "utf8mb4_0900_bin"), sql.NewRow("def", "mydb", "myview2", "SELECT * FROM myview WHERE i = 1", "NONE", "YES", "", "DEFINER", "utf8mb4", "utf8mb4_0900_bin"), }, }, { Query: "select table_name from information_schema.tables where table_schema = 'mydb' and table_type = 'VIEW' order by 1", Expected: []sql.Row{ sql.NewRow("myview"), sql.NewRow("myview2"), }, }, }
Functions ¶
This section is empty.
Types ¶
type GenericErrorQueryTest ¶
type GenericErrorQueryTest struct { Name string Query string Bindings map[string]sql.Expression }
GenericErrorQueryTest is a query test that is used to assert an error occurs for some query, without specifying what the error was.
type QueryErrorTest ¶
type QueryErrorTest struct { Query string Bindings map[string]sql.Expression ExpectedErr *errors.Kind ExpectedErrStr string }
type QueryPlanTest ¶
type QuickPrivilegeTest ¶
type QuickPrivilegeTest struct { Queries []string Expected []sql.Row ExpectedErr *errors.Kind ExpectingErr bool }
QuickPrivilegeTest specifically tests privileges on a predefined user (tester@localhost) using predefined tables and databases. Every test here can easily be represented by a UserPrivilegeTest, however this is intended to test specific privileges at a large scale, meaning there may be thousands of these tests, and hence the test data should be as small as possible.
All queries will be run as a root user with full privileges (intended for setup), with the last query running as the testing user (tester@localhost). For example, the first query may grant a SELECT privilege, while the second query is the SELECT query. Of note, the current database as set by the context is retained when switching from the root user to the test user. This does not mean that the test user automatically gains access to the database, but this is used for any queries that (incorrectly) only work with the current database.
ExpectingErr should be set when an error is expected, and it does not matter what the error is so long that it is one of the errors related to privilege checking (meaning a failed INSERT due to a missing column is NOT caught). If ExpectingErr is set and an error is given to ExpectedErr, then it is enforced that the error matches. However, if ExpectingErr is set and ExpectedErr is nil, then any privilege checking error will match.
Expected makes a distinction between the nil value and the empty value. A nil value means that we do not care about the result, only that it did not error (unless one of the error-asserting fields are set). A non-nil value asserts that the returned value matches our Expected value. If the returned value is nil, then we make a special case to match the non-nil empty row with it, due to the aforementioned distinction.
Statements that are run before every test (the state that all tests start with): CREATE TABLE mydb.test (pk BIGINT PRIMARY KEY, v1 BIGINT); CREATE TABLE mydb.test2 (pk BIGINT PRIMARY KEY, v1 BIGINT); CREATE TABLE otherdb.test (pk BIGINT PRIMARY KEY, v1 BIGINT); CREATE TABLE otherdb.test2 (pk BIGINT PRIMARY KEY, v1 BIGINT); INSERT INTO mydb.test VALUES (0, 0), (1, 1); INSERT INTO mydb.test2 VALUES (0, 1), (1, 2); INSERT INTO otherdb.test VALUES (1, 1), (2, 2); INSERT INTO otherdb.test2 VALUES (1, 1), (2, 2);
type ScriptTest ¶
type ScriptTest struct { // Name of the script test Name string // The sql statements to execute as setup, in order. Results are not checked, but statements must not error. SetUpScript []string // The set of assertions to make after setup, in order Assertions []ScriptTestAssertion // For tests that make a single assertion, Query can be set for the single assertion Query string // For tests that make a single assertion, Expected can be set for the single assertion Expected []sql.Row // For tests that make a single assertion, ExpectedErr can be set for the expected error ExpectedErr *errors.Kind // SkipPrepared is true when we skip a test for prepared statements only SkipPrepared bool }
type ScriptTestAssertion ¶
type ScriptTestAssertion struct { Query string Expected []sql.Row ExpectedErr *errors.Kind // ExpectedErrStr should be set for tests that expect a specific error string this is not linked to a custom error. // In most cases, errors should be linked to a custom error, however there are exceptions where this is not possible, // such as the use of the SIGNAL statement. ExpectedErrStr string // ExpectedWarning contains the expected warning code when a query generates warnings but not errors. ExpectedWarning int // ExpectedWarningsCount is used to test the expected number of warnings generated by a query. // The ExpectedWarning field must be set for warning counts to be checked. ExpectedWarningsCount int // ExpectedWarningMessageSubstring is used to test the contents of warning messages generated by a // query. The ExpectedWarning field must be set for warning messages to be checked. ExpectedWarningMessageSubstring string // SkipResultsCheck is used to skip assertions on expected Rows returned from a query. This should be used // sparingly, such as in cases where you only want to test warning messages. SkipResultsCheck bool // Bindings are variable mappings only used for prepared tests Bindings map[string]sql.Expression }
type ServerAuthenticationTest ¶
type ServerAuthenticationTest struct { Name string SetUpFunc func(ctx *sql.Context, t *testing.T, engine *sqle.Engine) SetUpScript []string Assertions []ServerAuthenticationTestAssertion }
ServerAuthenticationTest is used to define a test on the server authentication system. These tests always have the root account available, and the root account is used with any queries in the SetUpScript. The SetUpFunc is run before the SetUpScript.
type ServerAuthenticationTestAssertion ¶
type ServerAuthenticationTestAssertion struct { Username string Password string Query string ExpectedErr bool }
ServerAuthenticationTestAssertion is within a ServerAuthenticationTest to assert functionality.
type TransactionTest ¶
type TransactionTest struct { // Name of the script test Name string // The sql statements to execute as setup, in order. Results are not checked, but statements must not error. // Setup scripts are run as a distinct client separate from the client used in any assertions. SetUpScript []string // The set of assertions to make after setup, in order // The transaction test runner augments the ScriptTest runner by allowing embedding of a client string in a query // comment to name the client running the query, like so: // /* client a */ select * from myTable Assertions []ScriptTestAssertion }
TransactionTest is a script to test transaction correctness. It's similar to ScriptTest, but its assertions name clients that participate
type TypeWireTest ¶
TypeWireTest is used to ensure that types are properly represented over the wire (vs being directly returned from the engine).
type UserPrivilegeTest ¶
type UserPrivilegeTest struct { Name string SetUpScript []string Assertions []UserPrivilegeTestAssertion }
UserPrivilegeTest is used to define a test on the user and privilege systems. These tests always have the root account available, and the root account is used with any queries in the SetUpScript.
type UserPrivilegeTestAssertion ¶
type UserPrivilegeTestAssertion struct { User string Host string Query string Expected []sql.Row ExpectedErr *errors.Kind ExpectedErrStr string }
UserPrivilegeTestAssertion is within a UserPrivilegeTest to assert functionality.
type WriteQueryTest ¶
type WriteQueryTest struct { WriteQuery string ExpectedWriteResult []sql.Row SelectQuery string ExpectedSelect []sql.Row Bindings map[string]sql.Expression }
WriteQueryTest is a query test for INSERT, UPDATE, etc. statements. It has a query to run and a select query to validate the results.
Source Files
¶
- column_alias_queries.go
- create_table_queries.go
- delete_queries.go
- external_procedure_queries.go
- foreign_key_queries.go
- index_queries.go
- index_query_plans.go
- insert_queries.go
- join_queries.go
- json_scripts.go
- load_queries.go
- null_range_tests.go
- ordinal_ddl_queries.go
- priv_auth_queries.go
- procedure_queries.go
- queries.go
- query_plans.go
- replace_queries.go
- script_queries.go
- transaction_queries.go
- trigger_queries.go
- type_wire_queries.go
- update_queries.go
- variable_queries.go