mirror of
				https://github.com/pocoproject/poco.git
				synced 2025-10-26 18:42:41 +01:00 
			
		
		
		
	
		
			
				
	
	
		
			1139 lines
		
	
	
		
			48 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
			
		
		
	
	
			1139 lines
		
	
	
		
			48 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
| #include <climits>
 | ||
| #include <map>
 | ||
| 
 | ||
| #include "SQLParser.h"
 | ||
| #include "sql_asserts.h"
 | ||
| #include "thirdparty/microtest/microtest.h"
 | ||
| 
 | ||
| using namespace hsql;
 | ||
| 
 | ||
| TEST(SelectTest) {
 | ||
|   TEST_PARSE_SINGLE_SQL("SELECT * FROM students;", kStmtSelect, SelectStatement, result, stmt);
 | ||
| 
 | ||
|   ASSERT_NULL(stmt->whereClause);
 | ||
|   ASSERT_NULL(stmt->groupBy);
 | ||
| }
 | ||
| 
 | ||
| TEST(SelectExprTest) {
 | ||
|   TEST_PARSE_SINGLE_SQL("SELECT a, MAX(b), CUSTOM(c, F(un)) FROM students;", kStmtSelect, SelectStatement, result,
 | ||
|                         stmt);
 | ||
| 
 | ||
|   ASSERT_NULL(stmt->whereClause);
 | ||
|   ASSERT_NULL(stmt->groupBy);
 | ||
| 
 | ||
|   ASSERT_EQ(stmt->selectList->size(), 3);
 | ||
| 
 | ||
|   ASSERT(stmt->selectList->at(0)->isType(kExprColumnRef));
 | ||
|   ASSERT_STREQ(stmt->selectList->at(0)->getName(), "a");
 | ||
| 
 | ||
|   ASSERT(stmt->selectList->at(1)->isType(kExprFunctionRef));
 | ||
|   ASSERT_STREQ(stmt->selectList->at(1)->getName(), "MAX");
 | ||
|   ASSERT_NOTNULL(stmt->selectList->at(1)->exprList);
 | ||
|   ASSERT_EQ(stmt->selectList->at(1)->exprList->size(), 1);
 | ||
|   ASSERT(stmt->selectList->at(1)->exprList->at(0)->isType(kExprColumnRef));
 | ||
|   ASSERT_STREQ(stmt->selectList->at(1)->exprList->at(0)->getName(), "b");
 | ||
| 
 | ||
|   ASSERT(stmt->selectList->at(2)->isType(kExprFunctionRef));
 | ||
|   ASSERT_STREQ(stmt->selectList->at(2)->getName(), "CUSTOM");
 | ||
|   ASSERT_NOTNULL(stmt->selectList->at(2)->exprList);
 | ||
|   ASSERT_EQ(stmt->selectList->at(2)->exprList->size(), 2);
 | ||
|   ASSERT(stmt->selectList->at(2)->exprList->at(0)->isType(kExprColumnRef));
 | ||
|   ASSERT_STREQ(stmt->selectList->at(2)->exprList->at(0)->getName(), "c");
 | ||
| 
 | ||
|   ASSERT(stmt->selectList->at(2)->exprList->at(1)->isType(kExprFunctionRef));
 | ||
|   ASSERT_STREQ(stmt->selectList->at(2)->exprList->at(1)->getName(), "F");
 | ||
|   ASSERT_EQ(stmt->selectList->at(2)->exprList->at(1)->exprList->size(), 1);
 | ||
|   ASSERT(stmt->selectList->at(2)->exprList->at(1)->exprList->at(0)->isType(kExprColumnRef));
 | ||
|   ASSERT_STREQ(stmt->selectList->at(2)->exprList->at(1)->exprList->at(0)->getName(), "un");
 | ||
| }
 | ||
| 
 | ||
| TEST(SelectUnaryMinusTest) {
 | ||
|   TEST_PARSE_SINGLE_SQL(
 | ||
|       "SELECT 10 - 20, 10 + -20, 10 +-20, 10+-20, 9223372036854775807, -9223372036854775808, 10-5.2, 10+-5.2",
 | ||
|       kStmtSelect, SelectStatement, result, stmt);
 | ||
| 
 | ||
|   ASSERT_EQ(stmt->selectList->size(), 8);
 | ||
| 
 | ||
|   ASSERT_EQ(stmt->selectList->at(0)->type, kExprOperator);
 | ||
|   ASSERT_EQ(stmt->selectList->at(0)->opType, kOpMinus);
 | ||
|   ASSERT_EQ(stmt->selectList->at(0)->expr->type, kExprLiteralInt);
 | ||
|   ASSERT_EQ(stmt->selectList->at(0)->expr->ival, 10);
 | ||
|   ASSERT_EQ(stmt->selectList->at(0)->expr2->type, kExprLiteralInt);
 | ||
|   ASSERT_EQ(stmt->selectList->at(0)->expr2->ival, 20);
 | ||
| 
 | ||
|   ASSERT_EQ(stmt->selectList->at(1)->type, kExprOperator);
 | ||
|   ASSERT_EQ(stmt->selectList->at(1)->opType, kOpPlus);
 | ||
|   ASSERT_EQ(stmt->selectList->at(1)->expr->ival, 10);
 | ||
|   ASSERT_EQ(stmt->selectList->at(1)->expr2->type, kExprOperator);
 | ||
|   ASSERT_EQ(stmt->selectList->at(1)->expr2->opType, kOpUnaryMinus);
 | ||
|   ASSERT_EQ(stmt->selectList->at(1)->expr2->expr->type, kExprLiteralInt);
 | ||
|   ASSERT_EQ(stmt->selectList->at(1)->expr2->expr->ival, 20);
 | ||
| 
 | ||
|   ASSERT_EQ(stmt->selectList->at(2)->type, kExprOperator);
 | ||
|   ASSERT_EQ(stmt->selectList->at(2)->opType, kOpPlus);
 | ||
|   ASSERT_EQ(stmt->selectList->at(2)->expr->ival, 10);
 | ||
|   ASSERT_EQ(stmt->selectList->at(2)->expr2->type, kExprOperator);
 | ||
|   ASSERT_EQ(stmt->selectList->at(2)->expr2->opType, kOpUnaryMinus);
 | ||
|   ASSERT_EQ(stmt->selectList->at(2)->expr2->expr->type, kExprLiteralInt);
 | ||
|   ASSERT_EQ(stmt->selectList->at(2)->expr2->expr->ival, 20);
 | ||
| 
 | ||
|   ASSERT_EQ(stmt->selectList->at(3)->type, kExprOperator);
 | ||
|   ASSERT_EQ(stmt->selectList->at(3)->opType, kOpPlus);
 | ||
|   ASSERT_EQ(stmt->selectList->at(3)->expr->ival, 10);
 | ||
|   ASSERT_EQ(stmt->selectList->at(3)->expr2->type, kExprOperator);
 | ||
|   ASSERT_EQ(stmt->selectList->at(3)->expr2->opType, kOpUnaryMinus);
 | ||
|   ASSERT_EQ(stmt->selectList->at(3)->expr2->expr->type, kExprLiteralInt);
 | ||
|   ASSERT_EQ(stmt->selectList->at(3)->expr2->expr->ival, 20);
 | ||
| 
 | ||
|   ASSERT_EQ(stmt->selectList->at(4)->type, kExprLiteralInt);
 | ||
|   ASSERT_EQ(stmt->selectList->at(4)->ival, LLONG_MAX);
 | ||
| 
 | ||
|   ASSERT_EQ(stmt->selectList->at(5)->type, kExprLiteralInt);
 | ||
|   ASSERT_EQ(stmt->selectList->at(5)->ival, LLONG_MIN);
 | ||
| 
 | ||
|   ASSERT_EQ(stmt->selectList->at(6)->type, kExprOperator);
 | ||
|   ASSERT_EQ(stmt->selectList->at(6)->opType, kOpMinus);
 | ||
|   ASSERT_EQ(stmt->selectList->at(6)->expr->type, kExprLiteralInt);
 | ||
|   ASSERT_EQ(stmt->selectList->at(6)->expr->ival, 10);
 | ||
|   ASSERT_EQ(stmt->selectList->at(6)->expr2->type, kExprLiteralFloat);
 | ||
|   ASSERT_EQ(stmt->selectList->at(6)->expr2->fval, 5.2);
 | ||
| 
 | ||
|   ASSERT_EQ(stmt->selectList->at(7)->type, kExprOperator);
 | ||
|   ASSERT_EQ(stmt->selectList->at(7)->opType, kOpPlus);
 | ||
|   ASSERT_EQ(stmt->selectList->at(7)->expr->ival, 10);
 | ||
|   ASSERT_EQ(stmt->selectList->at(7)->expr2->type, kExprOperator);
 | ||
|   ASSERT_EQ(stmt->selectList->at(7)->expr2->opType, kOpUnaryMinus);
 | ||
|   ASSERT_EQ(stmt->selectList->at(7)->expr2->expr->type, kExprLiteralFloat);
 | ||
|   ASSERT_EQ(stmt->selectList->at(7)->expr2->expr->fval, 5.2);
 | ||
| }
 | ||
| 
 | ||
| TEST(SelectSubstrTest) {
 | ||
|   TEST_PARSE_SINGLE_SQL("SELECT SUBSTR(a, 3, 5) FROM students;", kStmtSelect, SelectStatement, result, stmt);
 | ||
| 
 | ||
|   ASSERT_NULL(stmt->whereClause);
 | ||
|   ASSERT_NULL(stmt->groupBy);
 | ||
| 
 | ||
|   ASSERT_EQ(stmt->selectList->size(), 1);
 | ||
| 
 | ||
|   ASSERT(stmt->selectList->at(0)->isType(kExprFunctionRef));
 | ||
|   ASSERT_STREQ(stmt->selectList->at(0)->getName(), "SUBSTR");
 | ||
| 
 | ||
|   ASSERT_NOTNULL(stmt->selectList->at(0)->exprList);
 | ||
|   ASSERT_EQ(stmt->selectList->at(0)->exprList->size(), 3);
 | ||
| 
 | ||
|   ASSERT(stmt->selectList->at(0)->exprList->at(0)->isType(kExprColumnRef));
 | ||
|   ASSERT_STREQ(stmt->selectList->at(0)->exprList->at(0)->getName(), "a");
 | ||
| 
 | ||
|   ASSERT(stmt->selectList->at(0)->exprList->at(1)->isType(kExprLiteralInt));
 | ||
|   ASSERT_EQ(stmt->selectList->at(0)->exprList->at(1)->ival, 3);
 | ||
| 
 | ||
|   ASSERT(stmt->selectList->at(0)->exprList->at(2)->isType(kExprLiteralInt));
 | ||
|   ASSERT_EQ(stmt->selectList->at(0)->exprList->at(2)->ival, 5);
 | ||
| }
 | ||
| 
 | ||
| TEST(SelectHavingTest) {
 | ||
|   TEST_PARSE_SINGLE_SQL("SELECT city, AVG(grade) AS avg_grade FROM students GROUP BY city HAVING AVG(grade) < -2.0",
 | ||
|                         kStmtSelect, SelectStatement, result, stmt);
 | ||
| 
 | ||
|   ASSERT_FALSE(stmt->selectDistinct);
 | ||
| 
 | ||
|   GroupByDescription* group = stmt->groupBy;
 | ||
|   ASSERT_NOTNULL(group);
 | ||
|   ASSERT_EQ(group->columns->size(), 1);
 | ||
|   ASSERT_EQ(group->having->opType, kOpLess);
 | ||
|   ASSERT(group->having->expr->isType(kExprFunctionRef));
 | ||
|   ASSERT(group->having->expr2->isType(kExprOperator));
 | ||
|   ASSERT_EQ(group->having->expr2->opType, kOpUnaryMinus);
 | ||
|   ASSERT_EQ(group->having->expr2->expr->fval, 2.0);
 | ||
| }
 | ||
| 
 | ||
| TEST(SelectDistinctTest) {
 | ||
|   TEST_PARSE_SINGLE_SQL("SELECT DISTINCT grade, city FROM students;", kStmtSelect, SelectStatement, result, stmt);
 | ||
| 
 | ||
|   ASSERT(stmt->selectDistinct);
 | ||
|   ASSERT_NULL(stmt->whereClause);
 | ||
| }
 | ||
| 
 | ||
| TEST(SelectSchemaTest) {
 | ||
|   TEST_PARSE_SINGLE_SQL("SELECT grade FROM some_schema.students;", kStmtSelect, SelectStatement, result, stmt);
 | ||
| 
 | ||
|   ASSERT(stmt->fromTable);
 | ||
|   ASSERT_EQ(std::string(stmt->fromTable->schema), "some_schema");
 | ||
| }
 | ||
| 
 | ||
| TEST(SelectGroupDistinctTest) {
 | ||
|   TEST_PARSE_SINGLE_SQL("SELECT city, COUNT(name), COUNT(DISTINCT grade) FROM students GROUP BY city;", kStmtSelect,
 | ||
|                         SelectStatement, result, stmt);
 | ||
| 
 | ||
|   ASSERT_FALSE(stmt->selectDistinct);
 | ||
|   ASSERT_EQ(stmt->selectList->size(), 3);
 | ||
|   ASSERT(!stmt->selectList->at(1)->distinct);
 | ||
|   ASSERT(stmt->selectList->at(2)->distinct);
 | ||
| }
 | ||
| 
 | ||
| TEST(OrderByTest) {
 | ||
|   TEST_PARSE_SINGLE_SQL("SELECT grade, city FROM students ORDER BY grade, city DESC;", kStmtSelect, SelectStatement,
 | ||
|                         result, stmt);
 | ||
| 
 | ||
|   ASSERT_NULL(stmt->whereClause);
 | ||
|   ASSERT_NOTNULL(stmt->order);
 | ||
| 
 | ||
|   ASSERT_EQ(stmt->order->size(), 2);
 | ||
|   ASSERT_EQ(stmt->order->at(0)->type, kOrderAsc);
 | ||
|   ASSERT_STREQ(stmt->order->at(0)->expr->name, "grade");
 | ||
| 
 | ||
|   ASSERT_EQ(stmt->order->at(1)->type, kOrderDesc);
 | ||
|   ASSERT_STREQ(stmt->order->at(1)->expr->name, "city");
 | ||
| }
 | ||
| 
 | ||
| TEST(SelectBetweenTest) {
 | ||
|   TEST_PARSE_SINGLE_SQL("SELECT grade, city FROM students WHERE grade BETWEEN 1 and c;", kStmtSelect, SelectStatement,
 | ||
|                         result, stmt);
 | ||
| 
 | ||
|   Expr* where = stmt->whereClause;
 | ||
|   ASSERT_NOTNULL(where);
 | ||
|   ASSERT(where->isType(kExprOperator));
 | ||
|   ASSERT_EQ(where->opType, kOpBetween);
 | ||
| 
 | ||
|   ASSERT_STREQ(where->expr->getName(), "grade");
 | ||
|   ASSERT(where->expr->isType(kExprColumnRef));
 | ||
| 
 | ||
|   ASSERT_EQ(where->exprList->size(), 2);
 | ||
|   ASSERT(where->exprList->at(0)->isType(kExprLiteralInt));
 | ||
|   ASSERT_EQ(where->exprList->at(0)->ival, 1);
 | ||
|   ASSERT(where->exprList->at(1)->isType(kExprColumnRef));
 | ||
|   ASSERT_STREQ(where->exprList->at(1)->getName(), "c");
 | ||
| }
 | ||
| 
 | ||
| TEST(SelectConditionalSelectTest) {
 | ||
|   TEST_PARSE_SINGLE_SQL(
 | ||
|       "SELECT * FROM t WHERE a = (SELECT MIN(v) FROM tt) AND EXISTS (SELECT * FROM test WHERE x < a);", kStmtSelect,
 | ||
|       SelectStatement, result, stmt);
 | ||
| 
 | ||
|   Expr* where = stmt->whereClause;
 | ||
|   ASSERT_NOTNULL(where);
 | ||
|   ASSERT(where->isType(kExprOperator));
 | ||
|   ASSERT_EQ(where->opType, kOpAnd);
 | ||
| 
 | ||
|   // a = (SELECT ...)
 | ||
|   Expr* cond1 = where->expr;
 | ||
|   ASSERT_NOTNULL(cond1);
 | ||
|   ASSERT_NOTNULL(cond1->expr);
 | ||
|   ASSERT_EQ(cond1->opType, kOpEquals);
 | ||
|   ASSERT_STREQ(cond1->expr->getName(), "a");
 | ||
|   ASSERT(cond1->expr->isType(kExprColumnRef));
 | ||
| 
 | ||
|   ASSERT_NOTNULL(cond1->expr2);
 | ||
|   ASSERT(cond1->expr2->isType(kExprSelect));
 | ||
| 
 | ||
|   SelectStatement* select2 = cond1->expr2->select;
 | ||
|   ASSERT_NOTNULL(select2);
 | ||
|   ASSERT_STREQ(select2->fromTable->getName(), "tt");
 | ||
| 
 | ||
|   // EXISTS (SELECT ...)
 | ||
|   Expr* cond2 = where->expr2;
 | ||
|   ASSERT_EQ(cond2->opType, kOpExists);
 | ||
|   ASSERT_NOTNULL(cond2->select);
 | ||
| 
 | ||
|   SelectStatement* ex_select = cond2->select;
 | ||
|   ASSERT_STREQ(ex_select->fromTable->getName(), "test");
 | ||
| }
 | ||
| 
 | ||
| TEST(SelectCaseWhen) {
 | ||
|   TEST_PARSE_SINGLE_SQL("SELECT MAX(CASE WHEN a = 'foo' THEN x ELSE 0 END) FROM test;", kStmtSelect, SelectStatement,
 | ||
|                         result, stmt);
 | ||
| 
 | ||
|   ASSERT_EQ(stmt->selectList->size(), 1);
 | ||
|   Expr* func = stmt->selectList->at(0);
 | ||
| 
 | ||
|   ASSERT_NOTNULL(func);
 | ||
|   ASSERT(func->isType(kExprFunctionRef));
 | ||
|   ASSERT_EQ(func->exprList->size(), 1);
 | ||
| 
 | ||
|   Expr* caseExpr = func->exprList->at(0);
 | ||
|   ASSERT_NOTNULL(caseExpr);
 | ||
|   ASSERT(caseExpr->isType(kExprOperator));
 | ||
|   ASSERT_EQ(caseExpr->opType, kOpCase);
 | ||
|   ASSERT_NULL(caseExpr->expr);
 | ||
|   ASSERT_NOTNULL(caseExpr->exprList);
 | ||
|   ASSERT_NOTNULL(caseExpr->expr2);
 | ||
|   ASSERT_EQ(caseExpr->exprList->size(), 1);
 | ||
|   ASSERT(caseExpr->expr2->isType(kExprLiteralInt));
 | ||
| 
 | ||
|   Expr* whenExpr = caseExpr->exprList->at(0);
 | ||
|   ASSERT(whenExpr->expr->isType(kExprOperator));
 | ||
|   ASSERT_EQ(whenExpr->expr->opType, kOpEquals);
 | ||
|   ASSERT(whenExpr->expr->expr->isType(kExprColumnRef));
 | ||
|   ASSERT(whenExpr->expr->expr2->isType(kExprLiteralString));
 | ||
| }
 | ||
| 
 | ||
| TEST(SelectCaseWhenWhen) {
 | ||
|   TEST_PARSE_SINGLE_SQL("SELECT CASE WHEN x = 1 THEN 1 WHEN 1.25 < x THEN 2 END FROM test;", kStmtSelect,
 | ||
|                         SelectStatement, result, stmt);
 | ||
| 
 | ||
|   ASSERT_EQ(stmt->selectList->size(), 1);
 | ||
|   Expr* caseExpr = stmt->selectList->at(0);
 | ||
|   ASSERT_NOTNULL(caseExpr);
 | ||
|   ASSERT(caseExpr->isType(kExprOperator));
 | ||
|   ASSERT_EQ(caseExpr->opType, kOpCase);
 | ||
|   // CASE [expr] [exprList]                                    [expr2]
 | ||
|   //             [expr]                  [expr]
 | ||
|   //             [expr]     [expr2]      [expr]        [expr2]
 | ||
|   // CASE (null) WHEN X = 1 THEN 1       WHEN 1.25 < x THEN 2  (null)
 | ||
|   ASSERT_NULL(caseExpr->expr);
 | ||
|   ASSERT_NOTNULL(caseExpr->exprList);
 | ||
|   ASSERT_NULL(caseExpr->expr2);
 | ||
|   ASSERT_EQ(caseExpr->exprList->size(), 2);
 | ||
| 
 | ||
|   Expr* whenExpr = caseExpr->exprList->at(0);
 | ||
|   ASSERT_EQ(whenExpr->expr->opType, kOpEquals);
 | ||
|   ASSERT(whenExpr->expr->expr->isType(kExprColumnRef));
 | ||
|   ASSERT(whenExpr->expr->expr2->isType(kExprLiteralInt));
 | ||
| 
 | ||
|   Expr* whenExpr2 = caseExpr->exprList->at(1);
 | ||
|   ASSERT_EQ(whenExpr2->expr->opType, kOpLess);
 | ||
|   ASSERT(whenExpr2->expr->expr->isType(kExprLiteralFloat));
 | ||
|   ASSERT(whenExpr2->expr->expr2->isType(kExprColumnRef));
 | ||
| }
 | ||
| 
 | ||
| TEST(SelectCaseValueWhenWhenElse) {
 | ||
|   TEST_PARSE_SINGLE_SQL("SELECT CASE x WHEN 1 THEN 0 WHEN 2 THEN 3 WHEN 8 THEN 7 ELSE 4 END FROM test;", kStmtSelect,
 | ||
|                         SelectStatement, result, stmt);
 | ||
| 
 | ||
|   ASSERT_EQ(stmt->selectList->size(), 1);
 | ||
|   Expr* caseExpr = stmt->selectList->at(0);
 | ||
|   ASSERT_NOTNULL(caseExpr);
 | ||
|   ASSERT(caseExpr->isType(kExprOperator));
 | ||
|   ASSERT_EQ(caseExpr->opType, kOpCase);
 | ||
|   ASSERT_NOTNULL(caseExpr->expr);
 | ||
|   ASSERT_NOTNULL(caseExpr->exprList);
 | ||
|   ASSERT_NOTNULL(caseExpr->expr2);
 | ||
|   ASSERT_EQ(caseExpr->exprList->size(), 3);
 | ||
|   ASSERT(caseExpr->expr->isType(kExprColumnRef));
 | ||
| 
 | ||
|   Expr* whenExpr = caseExpr->exprList->at(2);
 | ||
|   ASSERT(whenExpr->expr->isType(kExprLiteralInt));
 | ||
|   ASSERT_EQ(whenExpr->expr2->ival, 7);
 | ||
| }
 | ||
| 
 | ||
| TEST(SelectJoin) {
 | ||
|   TEST_PARSE_SINGLE_SQL(
 | ||
|       "SELECT City.name, Product.category, SUM(price) FROM fact\
 | ||
|       INNER JOIN City ON fact.city_id = City.id\
 | ||
|       OUTER JOIN Product ON fact.product_id = Product.id\
 | ||
|       GROUP BY City.name, Product.category;",
 | ||
|       kStmtSelect, SelectStatement, result, stmt);
 | ||
| 
 | ||
|   const TableRef* table = stmt->fromTable;
 | ||
|   const JoinDefinition* outer_join = table->join;
 | ||
|   ASSERT_EQ(table->type, kTableJoin);
 | ||
|   ASSERT_EQ(outer_join->type, kJoinFull);
 | ||
| 
 | ||
|   ASSERT_EQ(outer_join->right->type, kTableName);
 | ||
|   ASSERT_STREQ(outer_join->right->name, "Product");
 | ||
|   ASSERT_EQ(outer_join->condition->opType, kOpEquals);
 | ||
|   ASSERT_STREQ(outer_join->condition->expr->table, "fact");
 | ||
|   ASSERT_STREQ(outer_join->condition->expr->name, "product_id");
 | ||
|   ASSERT_STREQ(outer_join->condition->expr2->table, "Product");
 | ||
|   ASSERT_STREQ(outer_join->condition->expr2->name, "id");
 | ||
| 
 | ||
|   // Joins are are left associative.
 | ||
|   // So the second join should be on the left.
 | ||
|   ASSERT_EQ(outer_join->left->type, kTableJoin);
 | ||
| 
 | ||
|   const JoinDefinition* inner_join = outer_join->left->join;
 | ||
|   ASSERT_EQ(inner_join->type, kJoinInner);
 | ||
|   ASSERT_EQ(inner_join->left->type, kTableName);
 | ||
|   ASSERT_STREQ(inner_join->left->name, "fact");
 | ||
|   ASSERT_EQ(inner_join->right->type, kTableName);
 | ||
|   ASSERT_STREQ(inner_join->right->name, "City");
 | ||
| 
 | ||
|   ASSERT_EQ(inner_join->condition->opType, kOpEquals);
 | ||
|   ASSERT_STREQ(inner_join->condition->expr->table, "fact");
 | ||
|   ASSERT_STREQ(inner_join->condition->expr->name, "city_id");
 | ||
|   ASSERT_STREQ(inner_join->condition->expr2->table, "City");
 | ||
|   ASSERT_STREQ(inner_join->condition->expr2->name, "id");
 | ||
| }
 | ||
| 
 | ||
| TEST(SelectColumnOrder) {
 | ||
|   TEST_PARSE_SINGLE_SQL(
 | ||
|       "SELECT *\
 | ||
|     FROM a,\
 | ||
|          (SELECT a AS b FROM a) b,\
 | ||
|          (SELECT a AS c FROM a) c,\
 | ||
|          (SELECT a AS d FROM a) d;",
 | ||
|       kStmtSelect, SelectStatement, result, stmt);
 | ||
| 
 | ||
|   ASSERT_EQ(stmt->fromTable->list->size(), 4);
 | ||
| 
 | ||
|   // Make sure the order of the table list is corrects
 | ||
|   ASSERT_STREQ(stmt->fromTable->list->at(0)->name, "a");
 | ||
|   ASSERT_STREQ(stmt->fromTable->list->at(1)->alias->name, "b");
 | ||
|   ASSERT_STREQ(stmt->fromTable->list->at(2)->alias->name, "c");
 | ||
|   ASSERT_STREQ(stmt->fromTable->list->at(3)->alias->name, "d");
 | ||
| }
 | ||
| 
 | ||
| TEST(SelectAliasAbsent) {
 | ||
|   TEST_PARSE_SINGLE_SQL("SELECT * FROM students;", kStmtSelect, SelectStatement, result, stmt);
 | ||
| 
 | ||
|   ASSERT_NULL(stmt->fromTable->alias);
 | ||
| }
 | ||
| 
 | ||
| TEST(SelectAliasSimple) {
 | ||
|   TEST_PARSE_SINGLE_SQL("SELECT * FROM students AS s1;", kStmtSelect, SelectStatement, result, stmt);
 | ||
| 
 | ||
|   Alias* alias = stmt->fromTable->alias;
 | ||
|   ASSERT_NOTNULL(alias);
 | ||
|   ASSERT_STREQ(alias->name, "s1");
 | ||
|   ASSERT_NULL(alias->columns);
 | ||
| }
 | ||
| 
 | ||
| TEST(SelectAliasWithColumns) {
 | ||
|   TEST_PARSE_SINGLE_SQL("SELECT * FROM students AS s1(id, city);", kStmtSelect, SelectStatement, result, stmt);
 | ||
| 
 | ||
|   Alias* alias = stmt->fromTable->alias;
 | ||
|   ASSERT_NOTNULL(alias);
 | ||
| 
 | ||
|   ASSERT_NOTNULL(alias->name);
 | ||
|   ASSERT_STREQ(alias->name, "s1");
 | ||
| 
 | ||
|   ASSERT_NOTNULL(alias->columns);
 | ||
|   ASSERT_EQ(alias->columns->size(), 2);
 | ||
|   ASSERT_STREQ(alias->columns->at(0), "id");
 | ||
|   ASSERT_STREQ(alias->columns->at(1), "city");
 | ||
| }
 | ||
| 
 | ||
| TEST(SelectExpressionAlias) {
 | ||
|   TEST_PARSE_SINGLE_SQL("SELECT AVG(grade) avg_grade FROM students;", kStmtSelect, SelectStatement, result, stmt);
 | ||
| 
 | ||
|   ASSERT_NULL(stmt->fromTable->alias);
 | ||
|   ASSERT_EQ(stmt->selectList->size(), 1);
 | ||
|   ASSERT_TRUE(stmt->selectList->at(0)->isType(kExprFunctionRef));
 | ||
|   ASSERT_STREQ(stmt->selectList->at(0)->name, "AVG");
 | ||
|   ASSERT_TRUE(stmt->selectList->at(0)->hasAlias());
 | ||
|   ASSERT_STREQ(stmt->selectList->at(0)->alias, "avg_grade");
 | ||
| }
 | ||
| 
 | ||
| TEST(Operators) {
 | ||
|   SelectStatement* stmt;
 | ||
|   SQLParserResult result;
 | ||
| 
 | ||
|   SQLParser::parse(
 | ||
|       "SELECT * FROM foo where a =  1; \
 | ||
| 		    SELECT * FROM foo where a == 2; \
 | ||
| 		    SELECT * FROM foo where a != 1; \
 | ||
| 		    SELECT * FROM foo where a <> 1; \
 | ||
| 		    SELECT * FROM foo where a >  1; \
 | ||
| 		    SELECT * FROM foo where a <  1; \
 | ||
| 		    SELECT * FROM foo where a >= 1; \
 | ||
| 		    SELECT * FROM foo where a <= 1; \
 | ||
|         SELECT * FROM foo where a = TRUE; \
 | ||
|         SELECT * FROM foo where a = false;",
 | ||
|       &result);
 | ||
| 
 | ||
|   stmt = (SelectStatement*)result.getStatement(0);
 | ||
|   ASSERT_EQ(stmt->whereClause->opType, kOpEquals);
 | ||
|   ASSERT_EQ(stmt->whereClause->expr2->ival, 1);
 | ||
|   ASSERT_EQ(stmt->whereClause->expr2->isBoolLiteral, false);
 | ||
| 
 | ||
|   stmt = (SelectStatement*)result.getStatement(1);
 | ||
|   ASSERT_EQ(stmt->whereClause->opType, kOpEquals);
 | ||
|   ASSERT_EQ(stmt->whereClause->expr2->ival, 2);
 | ||
| 
 | ||
|   stmt = (SelectStatement*)result.getStatement(2);
 | ||
|   ASSERT_EQ(stmt->whereClause->opType, kOpNotEquals);
 | ||
| 
 | ||
|   stmt = (SelectStatement*)result.getStatement(3);
 | ||
|   ASSERT_EQ(stmt->whereClause->opType, kOpNotEquals);
 | ||
| 
 | ||
|   stmt = (SelectStatement*)result.getStatement(4);
 | ||
|   ASSERT_EQ(stmt->whereClause->opType, kOpGreater);
 | ||
| 
 | ||
|   stmt = (SelectStatement*)result.getStatement(5);
 | ||
|   ASSERT_EQ(stmt->whereClause->opType, kOpLess);
 | ||
| 
 | ||
|   stmt = (SelectStatement*)result.getStatement(6);
 | ||
|   ASSERT_EQ(stmt->whereClause->opType, kOpGreaterEq);
 | ||
| 
 | ||
|   stmt = (SelectStatement*)result.getStatement(7);
 | ||
|   ASSERT_EQ(stmt->whereClause->opType, kOpLessEq);
 | ||
| 
 | ||
|   stmt = (SelectStatement*)result.getStatement(8);
 | ||
|   ASSERT_EQ(stmt->whereClause->opType, kOpEquals);
 | ||
|   ASSERT_EQ(stmt->whereClause->expr2->ival, 1);
 | ||
|   ASSERT_EQ(stmt->whereClause->expr2->isBoolLiteral, true);
 | ||
| 
 | ||
|   stmt = (SelectStatement*)result.getStatement(9);
 | ||
|   ASSERT_EQ(stmt->whereClause->opType, kOpEquals);
 | ||
|   ASSERT_EQ(stmt->whereClause->expr2->ival, 0);
 | ||
|   ASSERT_EQ(stmt->whereClause->expr2->isBoolLiteral, true);
 | ||
| }
 | ||
| 
 | ||
| TEST(JoinTypes) {
 | ||
|   SelectStatement* stmt;
 | ||
|   SQLParserResult result;
 | ||
| 
 | ||
|   SQLParser::parse(
 | ||
|       "SELECT * FROM x join y on a=b; \
 | ||
| 		    SELECT * FROM x inner join y on a=b; \
 | ||
| 		    SELECT * FROM x left join y on a=b; \
 | ||
| 		    SELECT * FROM x left outer join y on a=b; \
 | ||
| 		    SELECT * FROM x right join y on a=b; \
 | ||
| 		    SELECT * FROM x right outer join y on a=b; \
 | ||
| 		    SELECT * FROM x full join y on a=b; \
 | ||
| 		    SELECT * FROM x outer join y on a=b; \
 | ||
| 		    SELECT * FROM x full outer join y on a=b; \
 | ||
| 		    SELECT * FROM x natural join y; \
 | ||
| 		    SELECT * FROM x cross join y on a=b; \
 | ||
| 		    SELECT * FROM x, y where a = b;",
 | ||
|       &result);
 | ||
| 
 | ||
|   stmt = (SelectStatement*)result.getStatement(0);
 | ||
|   ASSERT_EQ(stmt->fromTable->join->type, kJoinInner);
 | ||
| 
 | ||
|   stmt = (SelectStatement*)result.getStatement(1);
 | ||
|   ASSERT_EQ(stmt->fromTable->join->type, kJoinInner);
 | ||
| 
 | ||
|   stmt = (SelectStatement*)result.getStatement(2);
 | ||
|   ASSERT_EQ(stmt->fromTable->join->type, kJoinLeft);
 | ||
| 
 | ||
|   stmt = (SelectStatement*)result.getStatement(3);
 | ||
|   ASSERT_EQ(stmt->fromTable->join->type, kJoinLeft);
 | ||
| 
 | ||
|   stmt = (SelectStatement*)result.getStatement(4);
 | ||
|   ASSERT_EQ(stmt->fromTable->join->type, kJoinRight);
 | ||
| 
 | ||
|   stmt = (SelectStatement*)result.getStatement(5);
 | ||
|   ASSERT_EQ(stmt->fromTable->join->type, kJoinRight);
 | ||
| 
 | ||
|   stmt = (SelectStatement*)result.getStatement(6);
 | ||
|   ASSERT_EQ(stmt->fromTable->join->type, kJoinFull);
 | ||
| 
 | ||
|   stmt = (SelectStatement*)result.getStatement(7);
 | ||
|   ASSERT_EQ(stmt->fromTable->join->type, kJoinFull);
 | ||
| 
 | ||
|   stmt = (SelectStatement*)result.getStatement(8);
 | ||
|   ASSERT_EQ(stmt->fromTable->join->type, kJoinFull);
 | ||
| 
 | ||
|   stmt = (SelectStatement*)result.getStatement(9);
 | ||
|   ASSERT_EQ(stmt->fromTable->join->type, kJoinNatural);
 | ||
| 
 | ||
|   stmt = (SelectStatement*)result.getStatement(10);
 | ||
|   ASSERT_EQ(stmt->fromTable->join->type, kJoinCross);
 | ||
| 
 | ||
|   stmt = (SelectStatement*)result.getStatement(11);
 | ||
|   ASSERT_NULL(stmt->fromTable->join);
 | ||
| }
 | ||
| 
 | ||
| TEST(SetLimitOffset) {
 | ||
|   SelectStatement* stmt;
 | ||
| 
 | ||
|   TEST_PARSE_SQL_QUERY(
 | ||
|       "select a from t1 limit 1; \
 | ||
|                     select a from t1 limit 1 + 2; \
 | ||
|                     select a from t1 offset 1; \
 | ||
|                     select a from t1 offset 1 + 2; \
 | ||
|                     select a from t1 limit 1 offset 1; \
 | ||
|                     select a from t1 limit 1 + 2 offset 1 + 2; \
 | ||
|                     select a from t1 limit 1 offset NULL; \
 | ||
|                     select a from t1 limit ALL; \
 | ||
|                     select a from t1 limit NULL; \
 | ||
|                     select a from t1 limit ALL offset 1; \
 | ||
|                     select a from t1 limit NULL offset 1; \
 | ||
|                     select top 10 a from t1; \
 | ||
|                     select top 20 a from t1 limit 10; \
 | ||
|                     select a from t1 limit (SELECT MAX(b) FROM t1) offset (SELECT MIN(b) FROM t1);",
 | ||
|       result, 14);
 | ||
| 
 | ||
|   stmt = (SelectStatement*)result.getStatement(0);
 | ||
|   ASSERT_EQ(stmt->limit->limit->type, kExprLiteralInt);
 | ||
|   ASSERT_EQ(stmt->limit->limit->ival, 1);
 | ||
|   ASSERT_NULL(stmt->limit->offset);
 | ||
| 
 | ||
|   stmt = (SelectStatement*)result.getStatement(1);
 | ||
|   ASSERT_EQ(stmt->limit->limit->type, kExprOperator);
 | ||
|   ASSERT_EQ(stmt->limit->limit->opType, kOpPlus);
 | ||
|   ASSERT_EQ(stmt->limit->limit->expr->ival, 1);
 | ||
|   ASSERT_EQ(stmt->limit->limit->expr2->ival, 2);
 | ||
|   ASSERT_NULL(stmt->limit->offset);
 | ||
| 
 | ||
|   stmt = (SelectStatement*)result.getStatement(2);
 | ||
|   ASSERT_NULL(stmt->limit->limit);
 | ||
|   ASSERT_EQ(stmt->limit->offset->type, kExprLiteralInt);
 | ||
|   ASSERT_EQ(stmt->limit->offset->ival, 1);
 | ||
| 
 | ||
|   stmt = (SelectStatement*)result.getStatement(3);
 | ||
|   ASSERT_NULL(stmt->limit->limit);
 | ||
|   ASSERT_EQ(stmt->limit->offset->type, kExprOperator);
 | ||
|   ASSERT_EQ(stmt->limit->offset->opType, kOpPlus);
 | ||
|   ASSERT_EQ(stmt->limit->offset->expr->ival, 1);
 | ||
|   ASSERT_EQ(stmt->limit->offset->expr2->ival, 2);
 | ||
| 
 | ||
|   stmt = (SelectStatement*)result.getStatement(4);
 | ||
|   ASSERT_EQ(stmt->limit->limit->type, kExprLiteralInt);
 | ||
|   ASSERT_EQ(stmt->limit->limit->ival, 1);
 | ||
|   ASSERT_EQ(stmt->limit->offset->type, kExprLiteralInt);
 | ||
|   ASSERT_EQ(stmt->limit->offset->ival, 1);
 | ||
| 
 | ||
|   stmt = (SelectStatement*)result.getStatement(5);
 | ||
|   ASSERT_EQ(stmt->limit->limit->type, kExprOperator);
 | ||
|   ASSERT_EQ(stmt->limit->limit->opType, kOpPlus);
 | ||
|   ASSERT_EQ(stmt->limit->limit->expr->ival, 1);
 | ||
|   ASSERT_EQ(stmt->limit->limit->expr2->ival, 2);
 | ||
|   ASSERT_EQ(stmt->limit->offset->type, kExprOperator);
 | ||
|   ASSERT_EQ(stmt->limit->offset->opType, kOpPlus);
 | ||
|   ASSERT_EQ(stmt->limit->offset->expr->ival, 1);
 | ||
|   ASSERT_EQ(stmt->limit->offset->expr2->ival, 2);
 | ||
| 
 | ||
|   stmt = (SelectStatement*)result.getStatement(6);
 | ||
|   ASSERT_EQ(stmt->limit->limit->type, kExprLiteralInt);
 | ||
|   ASSERT_EQ(stmt->limit->limit->ival, 1);
 | ||
|   ASSERT_EQ(stmt->limit->offset->type, kExprLiteralNull);
 | ||
| 
 | ||
|   stmt = (SelectStatement*)result.getStatement(7);
 | ||
|   ASSERT_NULL(stmt->limit->limit);
 | ||
|   ASSERT_NULL(stmt->limit->offset);
 | ||
| 
 | ||
|   stmt = (SelectStatement*)result.getStatement(8);
 | ||
|   ASSERT_EQ(stmt->limit->limit->type, kExprLiteralNull);
 | ||
|   ASSERT_NULL(stmt->limit->offset);
 | ||
| 
 | ||
|   stmt = (SelectStatement*)result.getStatement(9);
 | ||
|   ASSERT_NULL(stmt->limit->limit);
 | ||
|   ASSERT_EQ(stmt->limit->offset->type, kExprLiteralInt);
 | ||
|   ASSERT_EQ(stmt->limit->offset->ival, 1);
 | ||
| 
 | ||
|   stmt = (SelectStatement*)result.getStatement(10);
 | ||
|   ASSERT_EQ(stmt->limit->limit->type, kExprLiteralNull);
 | ||
|   ASSERT_EQ(stmt->limit->offset->type, kExprLiteralInt);
 | ||
|   ASSERT_EQ(stmt->limit->offset->ival, 1);
 | ||
| 
 | ||
|   stmt = (SelectStatement*)result.getStatement(11);
 | ||
|   ASSERT_EQ(stmt->limit->limit->type, kExprLiteralInt);
 | ||
|   ASSERT_EQ(stmt->limit->limit->ival, 10);
 | ||
|   ASSERT_NULL(stmt->limit->offset);
 | ||
| 
 | ||
|   stmt = (SelectStatement*)result.getStatement(12);
 | ||
|   ASSERT_EQ(stmt->limit->limit->type, kExprLiteralInt);
 | ||
|   ASSERT_EQ(stmt->limit->limit->ival, 10);
 | ||
|   ASSERT_NULL(stmt->limit->offset);
 | ||
| 
 | ||
|   stmt = (SelectStatement*)result.getStatement(13);
 | ||
|   ASSERT_EQ(stmt->limit->limit->type, kExprSelect);
 | ||
|   ASSERT_EQ(stmt->limit->offset->type, kExprSelect);
 | ||
| }
 | ||
| 
 | ||
| TEST(Extract) {
 | ||
|   SelectStatement* stmt;
 | ||
| 
 | ||
|   TEST_PARSE_SQL_QUERY(
 | ||
|       "select extract(year from dc) FROM t;"
 | ||
|       "select x, extract(month from dc) AS t FROM t;"
 | ||
|       "select x FROM t WHERE extract(minute from dc) > 2011;",
 | ||
|       result, 3);
 | ||
| 
 | ||
|   stmt = (SelectStatement*)result.getStatement(0);
 | ||
|   ASSERT_TRUE(stmt->selectList);
 | ||
|   ASSERT_EQ(stmt->selectList->size(), 1u);
 | ||
|   ASSERT_EQ(stmt->selectList->at(0)->type, kExprExtract);
 | ||
|   ASSERT_EQ(stmt->selectList->at(0)->datetimeField, kDatetimeYear);
 | ||
|   ASSERT_TRUE(stmt->selectList->at(0)->expr);
 | ||
|   ASSERT_EQ(stmt->selectList->at(0)->expr->type, kExprColumnRef);
 | ||
| 
 | ||
|   stmt = (SelectStatement*)result.getStatement(1);
 | ||
|   ASSERT_TRUE(stmt->selectList);
 | ||
|   ASSERT_EQ(stmt->selectList->size(), 2u);
 | ||
|   ASSERT_EQ(stmt->selectList->at(1)->type, kExprExtract);
 | ||
|   ASSERT_EQ(stmt->selectList->at(1)->datetimeField, kDatetimeMonth);
 | ||
|   ASSERT_TRUE(stmt->selectList->at(1)->expr);
 | ||
|   ASSERT_EQ(stmt->selectList->at(1)->expr->type, kExprColumnRef);
 | ||
|   ASSERT_TRUE(stmt->selectList->at(1)->alias);
 | ||
|   ASSERT_EQ(stmt->selectList->at(1)->alias, std::string("t"));
 | ||
| 
 | ||
|   stmt = (SelectStatement*)result.getStatement(2);
 | ||
|   ASSERT_TRUE(stmt->whereClause);
 | ||
|   ASSERT_TRUE(stmt->whereClause->expr);
 | ||
|   ASSERT_EQ(stmt->whereClause->expr->type, kExprExtract);
 | ||
|   ASSERT_EQ(stmt->whereClause->expr->datetimeField, kDatetimeMinute);
 | ||
| }
 | ||
| 
 | ||
| TEST(CastExpression) {
 | ||
|   TEST_PARSE_SINGLE_SQL("SELECT CAST(10 AS INT);", kStmtSelect, SelectStatement, result, stmt);
 | ||
| 
 | ||
|   ASSERT_TRUE(stmt->selectList);
 | ||
|   ASSERT_FALSE(stmt->fromTable);
 | ||
|   ASSERT_FALSE(stmt->whereClause);
 | ||
|   ASSERT_FALSE(stmt->groupBy);
 | ||
| 
 | ||
|   ASSERT_EQ(stmt->selectList->size(), 1u);
 | ||
|   ASSERT_EQ(stmt->selectList->at(0)->type, kExprCast);
 | ||
|   ASSERT_EQ(stmt->selectList->at(0)->columnType, ColumnType(DataType::INT));
 | ||
|   ASSERT_EQ(stmt->selectList->at(0)->expr->type, kExprLiteralInt);
 | ||
| }
 | ||
| 
 | ||
| TEST(NoFromClause) {
 | ||
|   TEST_PARSE_SINGLE_SQL("SELECT 1 + 2;", kStmtSelect, SelectStatement, result, stmt);
 | ||
| 
 | ||
|   ASSERT_TRUE(stmt->selectList);
 | ||
|   ASSERT_FALSE(stmt->fromTable);
 | ||
|   ASSERT_FALSE(stmt->whereClause);
 | ||
|   ASSERT_FALSE(stmt->groupBy);
 | ||
| 
 | ||
|   ASSERT_EQ(stmt->selectList->size(), 1u);
 | ||
|   ASSERT_EQ(stmt->selectList->at(0)->type, kExprOperator);
 | ||
|   ASSERT_EQ(stmt->selectList->at(0)->expr->type, kExprLiteralInt);
 | ||
|   ASSERT_EQ(stmt->selectList->at(0)->expr2->type, kExprLiteralInt);
 | ||
| }
 | ||
| 
 | ||
| TEST(WithClauseSingle) {
 | ||
|   TEST_PARSE_SINGLE_SQL(
 | ||
|       "WITH "
 | ||
|       "a AS (SELECT name FROM peopleA)"
 | ||
|       "SELECT name FROM a;",
 | ||
|       kStmtSelect, SelectStatement, result, stmt)
 | ||
| 
 | ||
|   // with_description_list – count
 | ||
|   ASSERT_EQ(stmt->withDescriptions->size(), 1);
 | ||
| 
 | ||
|   // with_description – alias
 | ||
|   ASSERT_STREQ(stmt->withDescriptions->at(0)->alias, "a");
 | ||
| 
 | ||
|   // with_description – select stmt
 | ||
|   ASSERT_EQ(stmt->withDescriptions->at(0)->select->selectList->size(), 1)
 | ||
|   ASSERT_STREQ(stmt->withDescriptions->at(0)->select->selectList->at(0)->name, std::string("name"));
 | ||
|   ASSERT_STREQ(stmt->withDescriptions->at(0)->select->fromTable->name, std::string("peopleA"));
 | ||
| 
 | ||
|   // main select
 | ||
|   ASSERT_EQ(stmt->selectList->size(), 1);
 | ||
|   ASSERT_STREQ(stmt->selectList->at(0)->name, "name");
 | ||
|   ASSERT_STREQ(stmt->fromTable->name, "a");
 | ||
| }
 | ||
| 
 | ||
| TEST(WithClauseDouble) {
 | ||
|   TEST_PARSE_SINGLE_SQL(
 | ||
|       "WITH "
 | ||
|       "a AS (SELECT nameA FROM peopleA), "
 | ||
|       "b AS (SELECT nameB, cityB FROM peopleB) "
 | ||
|       "SELECT nameA FROM a;",
 | ||
|       kStmtSelect, SelectStatement, result, stmt)
 | ||
| 
 | ||
|   // with_description_list – count
 | ||
|   ASSERT_EQ(stmt->withDescriptions->size(), 2);
 | ||
| 
 | ||
|   // with_description – aliases
 | ||
|   ASSERT_STREQ(stmt->withDescriptions->at(0)->alias, "a");
 | ||
|   ASSERT_STREQ(stmt->withDescriptions->at(1)->alias, "b");
 | ||
| 
 | ||
|   // with_description – select stmts
 | ||
|   ASSERT_EQ(stmt->withDescriptions->at(0)->select->selectList->size(), 1)
 | ||
|   ASSERT_STREQ(stmt->withDescriptions->at(0)->select->fromTable->name, "peopleA");
 | ||
|   ASSERT_EQ(stmt->withDescriptions->at(1)->select->selectList->size(), 2)
 | ||
|   ASSERT_STREQ(stmt->withDescriptions->at(1)->select->fromTable->name, "peopleB");
 | ||
| 
 | ||
|   // main select
 | ||
|   ASSERT_EQ(stmt->selectList->size(), 1);
 | ||
|   ASSERT_STREQ(stmt->selectList->at(0)->name, "nameA");
 | ||
|   ASSERT_STREQ(stmt->fromTable->name, "a");
 | ||
| }
 | ||
| 
 | ||
| TEST(CastAsDate) {
 | ||
|   TEST_PARSE_SINGLE_SQL("SELECT CAST(ID AS DATE) FROM TEST", kStmtSelect, SelectStatement, result, stmt);
 | ||
| 
 | ||
|   ASSERT_TRUE(result.isValid());
 | ||
|   ASSERT_EQ(stmt->selectList->size(), 1);
 | ||
|   ASSERT_EQ(stmt->selectList->front()->type, kExprCast);
 | ||
|   ASSERT_EQ(stmt->selectList->front()->columnType.data_type, DataType::DATE);
 | ||
| }
 | ||
| 
 | ||
| TEST(DateLiteral) {
 | ||
|   TEST_PARSE_SINGLE_SQL("SELECT * FROM t WHERE a = DATE '1996-12-31'", kStmtSelect, SelectStatement, result, stmt);
 | ||
|   ASSERT_TRUE(result.isValid());
 | ||
|   stmt = (SelectStatement*)result.getStatement(0);
 | ||
|   ASSERT_EQ(stmt->whereClause->opType, kOpEquals);
 | ||
|   ASSERT_STREQ(stmt->whereClause->expr2->name, "1996-12-31");
 | ||
| }
 | ||
| 
 | ||
| TEST(IntervalLiteral) {
 | ||
|   SelectStatement* stmt;
 | ||
|   Expr* interval_literal;
 | ||
|   TEST_PARSE_SQL_QUERY(
 | ||
|       "SELECT a + 1 year FROM t;"
 | ||
|       "SELECT * FROM t where a = cast ('2000-01-01' AS DATE) - 30 days;",
 | ||
|       result, 2);
 | ||
| 
 | ||
|   stmt = (SelectStatement*)result.getStatement(0);
 | ||
|   ASSERT_TRUE(stmt->selectList);
 | ||
|   ASSERT_EQ(stmt->selectList->size(), 1u);
 | ||
|   ASSERT_EQ(stmt->selectList->at(0)->type, kExprOperator);
 | ||
|   ASSERT_TRUE(stmt->selectList->at(0)->expr2);
 | ||
|   interval_literal = stmt->selectList->at(0)->expr2;
 | ||
|   ASSERT_EQ(interval_literal->datetimeField, kDatetimeYear);
 | ||
|   ASSERT_EQ(interval_literal->ival, 1);
 | ||
|   ASSERT_EQ(interval_literal->type, kExprLiteralInterval);
 | ||
| 
 | ||
|   stmt = (SelectStatement*)result.getStatement(1);
 | ||
|   ASSERT_TRUE(stmt->whereClause);
 | ||
|   ASSERT_TRUE(stmt->whereClause->expr);
 | ||
|   ASSERT_TRUE(stmt->whereClause->type = kExprOperator);
 | ||
|   ASSERT_TRUE(stmt->whereClause->opType = kOpEquals);
 | ||
|   ASSERT_TRUE(stmt->whereClause->expr2);
 | ||
|   ASSERT_TRUE(stmt->whereClause->expr2->type = kExprOperator);
 | ||
|   ASSERT_TRUE(stmt->whereClause->expr2->opType = kOpPlus);
 | ||
|   ASSERT_TRUE(stmt->whereClause->expr2->expr2);
 | ||
|   interval_literal = stmt->whereClause->expr2->expr2;
 | ||
|   ASSERT_EQ(interval_literal->datetimeField, kDatetimeDay);
 | ||
|   ASSERT_EQ(interval_literal->ival, 30);
 | ||
|   ASSERT_EQ(interval_literal->type, kExprLiteralInterval);
 | ||
| 
 | ||
|   const auto interval_units = std::map<DatetimeField, std::string>{
 | ||
|       {kDatetimeSecond, "second"}, {kDatetimeMinute, "minute"}, {kDatetimeHour, "hour"},
 | ||
|       {kDatetimeDay, "day"},       {kDatetimeMonth, "month"},   {kDatetimeYear, "year"}};
 | ||
| 
 | ||
|   for (const auto& it : interval_units) {
 | ||
|     const auto& unit_string = it.second;
 | ||
|     const auto unit_string_plural = unit_string + "s";
 | ||
|     TEST_PARSE_SQL_QUERY("SELECT * FROM t WHERE a = 1 + 5 " + unit_string +
 | ||
|                              ";"
 | ||
|                              "SELECT * FROM t WHERE a = 1 + 5 " +
 | ||
|                              unit_string_plural +
 | ||
|                              ";"
 | ||
|                              "SELECT * FROM t WHERE a = 1 + INTERVAL '5'" +
 | ||
|                              unit_string +
 | ||
|                              ";"
 | ||
|                              "SELECT * FROM t WHERE a = 1 + INTERVAL '5 " +
 | ||
|                              unit_string +
 | ||
|                              "';"
 | ||
|                              "SELECT * FROM t WHERE a = 1 + INTERVAL '5 " +
 | ||
|                              unit_string_plural + "';",
 | ||
|                          result, 5);
 | ||
| 
 | ||
|     for (const auto& statement : result.getStatements()) {
 | ||
|       stmt = (SelectStatement*)statement;
 | ||
|       interval_literal = stmt->whereClause->expr2->expr2;
 | ||
|       ASSERT_EQ(interval_literal->datetimeField, it.first);
 | ||
|       ASSERT_EQ(interval_literal->ival, 5);
 | ||
|       ASSERT_EQ(interval_literal->type, kExprLiteralInterval);
 | ||
|     }
 | ||
|   }
 | ||
| }
 | ||
| 
 | ||
| TEST(LockingClauseWithoutWaitPolicy) {
 | ||
|   SelectStatement* stmt;
 | ||
|   TEST_PARSE_SQL_QUERY(
 | ||
|       "SELECT * FROM t WHERE a = 10 FOR UPDATE;"
 | ||
|       "SELECT * FROM t WHERE a = 10 FOR SHARE;"
 | ||
|       "SELECT * FROM t WHERE a = 10 FOR NO KEY UPDATE FOR KEY SHARE;",
 | ||
|       result, 3);
 | ||
| 
 | ||
|   stmt = (SelectStatement*)result.getStatement(0);
 | ||
|   ASSERT_EQ(stmt->lockings->size(), 1);
 | ||
|   ASSERT_FALSE(stmt->lockings->at(0)->tables);
 | ||
|   ASSERT_EQ(stmt->lockings->at(0)->rowLockMode, RowLockMode::ForUpdate);
 | ||
|   ASSERT_EQ(stmt->lockings->at(0)->rowLockWaitPolicy, RowLockWaitPolicy::None);
 | ||
| 
 | ||
|   stmt = (SelectStatement*)result.getStatement(1);
 | ||
|   ASSERT_EQ(stmt->lockings->size(), 1);
 | ||
|   ASSERT_FALSE(stmt->lockings->at(0)->tables);
 | ||
|   ASSERT_EQ(stmt->lockings->at(0)->rowLockMode, RowLockMode::ForShare);
 | ||
|   ASSERT_EQ(stmt->lockings->at(0)->rowLockWaitPolicy, RowLockWaitPolicy::None);
 | ||
| 
 | ||
|   stmt = (SelectStatement*)result.getStatement(2);
 | ||
|   ASSERT_EQ(stmt->lockings->size(), 2);
 | ||
|   ASSERT_FALSE(stmt->lockings->at(0)->tables);
 | ||
|   ASSERT_FALSE(stmt->lockings->at(1)->tables);
 | ||
|   ASSERT_EQ(stmt->lockings->at(0)->rowLockMode, RowLockMode::ForNoKeyUpdate);
 | ||
|   ASSERT_EQ(stmt->lockings->at(0)->rowLockWaitPolicy, RowLockWaitPolicy::None);
 | ||
|   ASSERT_EQ(stmt->lockings->at(1)->rowLockMode, RowLockMode::ForKeyShare);
 | ||
|   ASSERT_EQ(stmt->lockings->at(1)->rowLockWaitPolicy, RowLockWaitPolicy::None);
 | ||
| }
 | ||
| 
 | ||
| TEST(LockingClauseWithWaitPolicy) {
 | ||
|   SelectStatement* stmt;
 | ||
|   TEST_PARSE_SQL_QUERY(
 | ||
|       "SELECT * FROM t WHERE a = 10 FOR UPDATE NOWAIT;"
 | ||
|       "SELECT * FROM t WHERE a = 10 FOR SHARE NOWAIT;"
 | ||
|       "SELECT * FROM t WHERE a = 10 FOR NO KEY UPDATE NOWAIT;"
 | ||
|       "SELECT * FROM t WHERE a = 10 FOR KEY SHARE NOWAIT;"
 | ||
|       "SELECT * FROM t WHERE a = 10 FOR UPDATE SKIP LOCKED;"
 | ||
|       "SELECT * FROM t WHERE a = 10 FOR SHARE SKIP LOCKED;"
 | ||
|       "SELECT * FROM t WHERE a = 10 FOR NO KEY UPDATE SKIP LOCKED;"
 | ||
|       "SELECT * FROM t WHERE a = 10 FOR KEY SHARE SKIP LOCKED;",
 | ||
|       result, 8);
 | ||
| 
 | ||
|   stmt = (SelectStatement*)result.getStatement(0);
 | ||
|   ASSERT_EQ(stmt->lockings->size(), 1);
 | ||
|   ASSERT_FALSE(stmt->lockings->at(0)->tables);
 | ||
|   ASSERT_EQ(stmt->lockings->at(0)->rowLockMode, RowLockMode::ForUpdate);
 | ||
|   ASSERT_EQ(stmt->lockings->at(0)->rowLockWaitPolicy, RowLockWaitPolicy::NoWait);
 | ||
| 
 | ||
|   stmt = (SelectStatement*)result.getStatement(1);
 | ||
|   ASSERT_EQ(stmt->lockings->size(), 1);
 | ||
|   ASSERT_FALSE(stmt->lockings->at(0)->tables);
 | ||
|   ASSERT_EQ(stmt->lockings->at(0)->rowLockMode, RowLockMode::ForShare);
 | ||
|   ASSERT_EQ(stmt->lockings->at(0)->rowLockWaitPolicy, RowLockWaitPolicy::NoWait);
 | ||
| 
 | ||
|   stmt = (SelectStatement*)result.getStatement(2);
 | ||
|   ASSERT_EQ(stmt->lockings->size(), 1);
 | ||
|   ASSERT_FALSE(stmt->lockings->at(0)->tables);
 | ||
|   ASSERT_EQ(stmt->lockings->at(0)->rowLockMode, RowLockMode::ForNoKeyUpdate);
 | ||
|   ASSERT_EQ(stmt->lockings->at(0)->rowLockWaitPolicy, RowLockWaitPolicy::NoWait);
 | ||
| 
 | ||
|   stmt = (SelectStatement*)result.getStatement(3);
 | ||
|   ASSERT_EQ(stmt->lockings->size(), 1);
 | ||
|   ASSERT_FALSE(stmt->lockings->at(0)->tables);
 | ||
|   ASSERT_EQ(stmt->lockings->at(0)->rowLockMode, RowLockMode::ForKeyShare);
 | ||
|   ASSERT_EQ(stmt->lockings->at(0)->rowLockWaitPolicy, RowLockWaitPolicy::NoWait);
 | ||
| 
 | ||
|   stmt = (SelectStatement*)result.getStatement(4);
 | ||
|   ASSERT_EQ(stmt->lockings->size(), 1);
 | ||
|   ASSERT_FALSE(stmt->lockings->at(0)->tables);
 | ||
|   ASSERT_EQ(stmt->lockings->at(0)->rowLockMode, RowLockMode::ForUpdate);
 | ||
|   ASSERT_EQ(stmt->lockings->at(0)->rowLockWaitPolicy, RowLockWaitPolicy::SkipLocked);
 | ||
| 
 | ||
|   stmt = (SelectStatement*)result.getStatement(5);
 | ||
|   ASSERT_EQ(stmt->lockings->size(), 1);
 | ||
|   ASSERT_FALSE(stmt->lockings->at(0)->tables);
 | ||
|   ASSERT_EQ(stmt->lockings->at(0)->rowLockMode, RowLockMode::ForShare);
 | ||
|   ASSERT_EQ(stmt->lockings->at(0)->rowLockWaitPolicy, RowLockWaitPolicy::SkipLocked);
 | ||
| 
 | ||
|   stmt = (SelectStatement*)result.getStatement(6);
 | ||
|   ASSERT_EQ(stmt->lockings->size(), 1);
 | ||
|   ASSERT_FALSE(stmt->lockings->at(0)->tables);
 | ||
|   ASSERT_EQ(stmt->lockings->at(0)->rowLockMode, RowLockMode::ForNoKeyUpdate);
 | ||
|   ASSERT_EQ(stmt->lockings->at(0)->rowLockWaitPolicy, RowLockWaitPolicy::SkipLocked);
 | ||
| 
 | ||
|   stmt = (SelectStatement*)result.getStatement(7);
 | ||
|   ASSERT_EQ(stmt->lockings->size(), 1);
 | ||
|   ASSERT_FALSE(stmt->lockings->at(0)->tables);
 | ||
|   ASSERT_EQ(stmt->lockings->at(0)->rowLockMode, RowLockMode::ForKeyShare);
 | ||
|   ASSERT_EQ(stmt->lockings->at(0)->rowLockWaitPolicy, RowLockWaitPolicy::SkipLocked);
 | ||
| }
 | ||
| 
 | ||
| TEST(LockingClauseWithTableReference) {
 | ||
|   SelectStatement* stmt;
 | ||
|   TEST_PARSE_SQL_QUERY(
 | ||
|       "SELECT * FROM t WHERE a = 10 FOR UPDATE OF t;"
 | ||
|       "SELECT * FROM t, c WHERE t.a = 10 FOR SHARE OF t,c;"
 | ||
|       "SELECT * FROM t, c WHERE t.a = 10 FOR NO KEY UPDATE OF t,c NOWAIT;",
 | ||
|       result, 3);
 | ||
| 
 | ||
|   stmt = (SelectStatement*)result.getStatement(0);
 | ||
|   ASSERT_EQ(stmt->lockings->size(), 1);
 | ||
|   ASSERT_EQ(stmt->lockings->at(0)->rowLockMode, RowLockMode::ForUpdate);
 | ||
|   ASSERT_EQ(stmt->lockings->at(0)->rowLockWaitPolicy, RowLockWaitPolicy::None);
 | ||
|   ASSERT_EQ(stmt->lockings->at(0)->tables->size(), 1);
 | ||
|   ASSERT_STREQ(stmt->lockings->at(0)->tables->at(0), "t");
 | ||
| 
 | ||
|   stmt = (SelectStatement*)result.getStatement(1);
 | ||
|   ASSERT_EQ(stmt->lockings->size(), 1);
 | ||
|   ASSERT_EQ(stmt->lockings->at(0)->rowLockMode, RowLockMode::ForShare);
 | ||
|   ASSERT_EQ(stmt->lockings->at(0)->rowLockWaitPolicy, RowLockWaitPolicy::None);
 | ||
|   ASSERT_EQ(stmt->lockings->at(0)->tables->size(), 2);
 | ||
|   ASSERT_STREQ(stmt->lockings->at(0)->tables->at(0), "t");
 | ||
|   ASSERT_STREQ(stmt->lockings->at(0)->tables->at(1), "c");
 | ||
| 
 | ||
|   stmt = (SelectStatement*)result.getStatement(2);
 | ||
|   ASSERT_EQ(stmt->lockings->size(), 1);
 | ||
|   ASSERT_EQ(stmt->lockings->at(0)->rowLockMode, RowLockMode::ForNoKeyUpdate);
 | ||
|   ASSERT_EQ(stmt->lockings->at(0)->rowLockWaitPolicy, RowLockWaitPolicy::NoWait);
 | ||
|   ASSERT_EQ(stmt->lockings->at(0)->tables->size(), 2);
 | ||
|   ASSERT_STREQ(stmt->lockings->at(0)->tables->at(0), "t");
 | ||
|   ASSERT_STREQ(stmt->lockings->at(0)->tables->at(1), "c");
 | ||
| }
 | ||
| 
 | ||
| TEST(MultipleLockingClause) {
 | ||
|   SelectStatement* stmt;
 | ||
|   TEST_PARSE_SQL_QUERY(
 | ||
|       "SELECT * FROM t, c WHERE t.a = 10 FOR NO KEY UPDATE OF t FOR KEY SHARE OF c;"
 | ||
|       "SELECT * FROM t, c WHERE t.a = 10 FOR SHARE OF t SKIP LOCKED FOR UPDATE OF c NOWAIT;"
 | ||
|       "SELECT * FROM t, c, s WHERE t.a = 10 FOR NO KEY UPDATE FOR SHARE OF t SKIP LOCKED FOR UPDATE OF c, s NOWAIT;",
 | ||
|       result, 3);
 | ||
| 
 | ||
|   stmt = (SelectStatement*)result.getStatement(0);
 | ||
|   ASSERT_EQ(stmt->lockings->size(), 2);
 | ||
|   ASSERT_EQ(stmt->lockings->at(0)->rowLockMode, RowLockMode::ForNoKeyUpdate);
 | ||
|   ASSERT_EQ(stmt->lockings->at(0)->rowLockWaitPolicy, RowLockWaitPolicy::None);
 | ||
|   ASSERT_EQ(stmt->lockings->at(0)->tables->size(), 1);
 | ||
|   ASSERT_STREQ(stmt->lockings->at(0)->tables->at(0), "t");
 | ||
|   ASSERT_EQ(stmt->lockings->at(1)->rowLockMode, RowLockMode::ForKeyShare);
 | ||
|   ASSERT_EQ(stmt->lockings->at(1)->rowLockWaitPolicy, RowLockWaitPolicy::None);
 | ||
|   ASSERT_EQ(stmt->lockings->at(1)->tables->size(), 1);
 | ||
|   ASSERT_STREQ(stmt->lockings->at(1)->tables->at(0), "c");
 | ||
| 
 | ||
|   stmt = (SelectStatement*)result.getStatement(1);
 | ||
|   ASSERT_EQ(stmt->lockings->size(), 2);
 | ||
|   ASSERT_EQ(stmt->lockings->at(0)->rowLockMode, RowLockMode::ForShare);
 | ||
|   ASSERT_EQ(stmt->lockings->at(0)->rowLockWaitPolicy, RowLockWaitPolicy::SkipLocked);
 | ||
|   ASSERT_EQ(stmt->lockings->at(0)->tables->size(), 1);
 | ||
|   ASSERT_STREQ(stmt->lockings->at(0)->tables->at(0), "t");
 | ||
|   ASSERT_EQ(stmt->lockings->at(1)->rowLockMode, RowLockMode::ForUpdate);
 | ||
|   ASSERT_EQ(stmt->lockings->at(1)->rowLockWaitPolicy, RowLockWaitPolicy::NoWait);
 | ||
|   ASSERT_EQ(stmt->lockings->at(0)->tables->size(), 1);
 | ||
|   ASSERT_STREQ(stmt->lockings->at(1)->tables->at(0), "c");
 | ||
| 
 | ||
|   stmt = (SelectStatement*)result.getStatement(2);
 | ||
|   ASSERT_EQ(stmt->lockings->size(), 3);
 | ||
|   ASSERT_EQ(stmt->lockings->at(0)->rowLockMode, RowLockMode::ForNoKeyUpdate);
 | ||
|   ASSERT_EQ(stmt->lockings->at(0)->rowLockWaitPolicy, RowLockWaitPolicy::None);
 | ||
|   ASSERT_FALSE(stmt->lockings->at(0)->tables);
 | ||
| 
 | ||
|   ASSERT_EQ(stmt->lockings->at(1)->rowLockMode, RowLockMode::ForShare);
 | ||
|   ASSERT_EQ(stmt->lockings->at(1)->rowLockWaitPolicy, RowLockWaitPolicy::SkipLocked);
 | ||
|   ASSERT_EQ(stmt->lockings->at(1)->tables->size(), 1);
 | ||
|   ASSERT_STREQ(stmt->lockings->at(1)->tables->at(0), "t");
 | ||
| 
 | ||
|   ASSERT_EQ(stmt->lockings->at(2)->rowLockMode, RowLockMode::ForUpdate);
 | ||
|   ASSERT_EQ(stmt->lockings->at(2)->rowLockWaitPolicy, RowLockWaitPolicy::NoWait);
 | ||
|   ASSERT_EQ(stmt->lockings->at(2)->tables->size(), 2);
 | ||
|   ASSERT_STREQ(stmt->lockings->at(2)->tables->at(0), "c");
 | ||
|   ASSERT_STREQ(stmt->lockings->at(2)->tables->at(1), "s");
 | ||
| }
 | ||
| 
 | ||
| TEST(WindowFunctions) {
 | ||
|   SelectStatement* stmt;
 | ||
|   TEST_PARSE_SQL_QUERY(
 | ||
|       "SELECT t2, 1 / avg(t1) OVER(), rank() OVER(ORDER BY t1 DESC) rnk FROM t;"
 | ||
|       "SELECT avg(t1) OVER(PARTITION BY t2, t3 ORDER BY t4, t5 ROWS UNBOUNDED PRECEDING) FROM t;"
 | ||
|       "SELECT rank() OVER(PARTITION BY t1 ORDER BY t2 ROWS BETWEEN 25 PRECEDING AND 2 FOLLOWING) FROM t;"
 | ||
|       "SELECT rank() OVER(PARTITION BY t1 ORDER BY t2 RANGE BETWEEN UNBOUNDED PRECEDING AND UNBOUNDED FOLLOWING) FROM "
 | ||
|       "t;"
 | ||
|       "SELECT rank() OVER(PARTITION BY t1 ORDER BY t2 GROUPS BETWEEN UNBOUNDED PRECEDING AND CURRENT ROW) FROM t;",
 | ||
|       result, 5);
 | ||
| 
 | ||
|   stmt = (SelectStatement*)result.getStatement(0);
 | ||
|   ASSERT_TRUE(stmt->selectList);
 | ||
|   ASSERT_EQ(stmt->selectList->size(), 3);
 | ||
|   ASSERT_TRUE(stmt->fromTable);
 | ||
|   ASSERT_EQ(stmt->fromTable->type, kTableName);
 | ||
|   ASSERT_STREQ(stmt->fromTable->name, "t");
 | ||
| 
 | ||
|   ASSERT_EQ(stmt->selectList->at(1)->type, kExprOperator);
 | ||
|   ASSERT_EQ(stmt->selectList->at(1)->opType, kOpSlash);
 | ||
|   ASSERT_TRUE(stmt->selectList->at(1)->expr);
 | ||
|   ASSERT_EQ(stmt->selectList->at(1)->expr->type, kExprLiteralInt);
 | ||
|   ASSERT_EQ(stmt->selectList->at(1)->expr->ival, 1);
 | ||
| 
 | ||
|   ASSERT_TRUE(stmt->selectList->at(1)->expr2);
 | ||
|   ASSERT_EQ(stmt->selectList->at(1)->expr2->type, kExprFunctionRef);
 | ||
|   ASSERT_STREQ(stmt->selectList->at(1)->expr2->name, "avg");
 | ||
|   ASSERT_TRUE(stmt->selectList->at(1)->expr2->exprList);
 | ||
|   ASSERT_EQ(stmt->selectList->at(1)->expr2->exprList->size(), 1);
 | ||
|   ASSERT_EQ(stmt->selectList->at(1)->expr2->exprList->at(0)->type, kExprColumnRef);
 | ||
|   ASSERT_STREQ(stmt->selectList->at(1)->expr2->exprList->at(0)->name, "t1");
 | ||
| 
 | ||
|   ASSERT_TRUE(stmt->selectList->at(1)->expr2->windowDescription);
 | ||
|   ASSERT_FALSE(stmt->selectList->at(1)->expr2->windowDescription->partitionList);
 | ||
|   ASSERT_FALSE(stmt->selectList->at(1)->expr2->windowDescription->orderList);
 | ||
|   ASSERT_TRUE(stmt->selectList->at(1)->expr2->windowDescription->frameDescription);
 | ||
|   ASSERT_EQ(stmt->selectList->at(1)->expr2->windowDescription->frameDescription->type, kRange);
 | ||
|   ASSERT_TRUE(stmt->selectList->at(1)->expr2->windowDescription->frameDescription->start);
 | ||
|   ASSERT_EQ(stmt->selectList->at(1)->expr2->windowDescription->frameDescription->start->offset, 0);
 | ||
|   ASSERT_EQ(stmt->selectList->at(1)->expr2->windowDescription->frameDescription->start->type, kPreceding);
 | ||
|   ASSERT_TRUE(stmt->selectList->at(1)->expr2->windowDescription->frameDescription->start->unbounded);
 | ||
|   ASSERT_TRUE(stmt->selectList->at(1)->expr2->windowDescription->frameDescription->end);
 | ||
|   ASSERT_EQ(stmt->selectList->at(1)->expr2->windowDescription->frameDescription->end->offset, 0);
 | ||
|   ASSERT_EQ(stmt->selectList->at(1)->expr2->windowDescription->frameDescription->end->type, kCurrentRow);
 | ||
|   ASSERT_FALSE(stmt->selectList->at(1)->expr2->windowDescription->frameDescription->end->unbounded);
 | ||
| 
 | ||
|   ASSERT_TRUE(stmt->selectList->at(2));
 | ||
|   ASSERT_EQ(stmt->selectList->at(2)->type, kExprFunctionRef);
 | ||
|   ASSERT_STREQ(stmt->selectList->at(2)->name, "rank");
 | ||
|   ASSERT_TRUE(stmt->selectList->at(2)->alias);
 | ||
|   ASSERT_STREQ(stmt->selectList->at(2)->alias, "rnk");
 | ||
|   ASSERT_TRUE(stmt->selectList->at(2)->exprList);
 | ||
|   ASSERT_TRUE(stmt->selectList->at(2)->exprList->empty());
 | ||
| 
 | ||
|   ASSERT_TRUE(stmt->selectList->at(2)->windowDescription);
 | ||
|   ASSERT_FALSE(stmt->selectList->at(2)->windowDescription->partitionList);
 | ||
|   ASSERT_TRUE(stmt->selectList->at(2)->windowDescription->orderList);
 | ||
|   ASSERT_EQ(stmt->selectList->at(2)->windowDescription->orderList->size(), 1);
 | ||
|   ASSERT_EQ(stmt->selectList->at(2)->windowDescription->orderList->at(0)->type, kOrderDesc);
 | ||
|   ASSERT_TRUE(stmt->selectList->at(2)->windowDescription->orderList->at(0)->expr);
 | ||
|   ASSERT_EQ(stmt->selectList->at(2)->windowDescription->orderList->at(0)->expr->type, kExprColumnRef);
 | ||
|   ASSERT_STREQ(stmt->selectList->at(2)->windowDescription->orderList->at(0)->expr->name, "t1");
 | ||
|   ASSERT_TRUE(stmt->selectList->at(2)->windowDescription->frameDescription);
 | ||
| 
 | ||
|   stmt = (SelectStatement*)result.getStatement(1);
 | ||
|   ASSERT_TRUE(stmt->selectList);
 | ||
|   ASSERT_EQ(stmt->selectList->size(), 1);
 | ||
|   ASSERT_TRUE(stmt->fromTable);
 | ||
|   ASSERT_EQ(stmt->fromTable->type, kTableName);
 | ||
|   ASSERT_STREQ(stmt->fromTable->name, "t");
 | ||
| 
 | ||
|   ASSERT_EQ(stmt->selectList->at(0)->type, kExprFunctionRef);
 | ||
|   ASSERT_STREQ(stmt->selectList->at(0)->name, "avg");
 | ||
|   ASSERT_EQ(stmt->selectList->at(0)->exprList->size(), 1);
 | ||
|   ASSERT_STREQ(stmt->selectList->at(0)->exprList->at(0)->name, "t1");
 | ||
| 
 | ||
|   ASSERT_TRUE(stmt->selectList->at(0)->windowDescription);
 | ||
|   ASSERT_TRUE(stmt->selectList->at(0)->windowDescription->partitionList);
 | ||
|   ASSERT_EQ(stmt->selectList->at(0)->windowDescription->partitionList->size(), 2);
 | ||
|   ASSERT_EQ(stmt->selectList->at(0)->windowDescription->partitionList->at(0)->type, kExprColumnRef);
 | ||
|   ASSERT_STREQ(stmt->selectList->at(0)->windowDescription->partitionList->at(0)->name, "t2");
 | ||
|   ASSERT_EQ(stmt->selectList->at(0)->windowDescription->partitionList->at(1)->type, kExprColumnRef);
 | ||
|   ASSERT_STREQ(stmt->selectList->at(0)->windowDescription->partitionList->at(1)->name, "t3");
 | ||
| 
 | ||
|   ASSERT_TRUE(stmt->selectList->at(0)->windowDescription->orderList);
 | ||
|   ASSERT_EQ(stmt->selectList->at(0)->windowDescription->orderList->size(), 2);
 | ||
|   ASSERT_EQ(stmt->selectList->at(0)->windowDescription->orderList->at(0)->type, kOrderAsc);
 | ||
|   ASSERT_TRUE(stmt->selectList->at(0)->windowDescription->orderList->at(0)->expr);
 | ||
|   ASSERT_EQ(stmt->selectList->at(0)->windowDescription->orderList->at(0)->expr->type, kExprColumnRef);
 | ||
|   ASSERT_STREQ(stmt->selectList->at(0)->windowDescription->orderList->at(0)->expr->name, "t4");
 | ||
|   ASSERT_EQ(stmt->selectList->at(0)->windowDescription->orderList->at(1)->expr->type, kExprColumnRef);
 | ||
|   ASSERT_STREQ(stmt->selectList->at(0)->windowDescription->orderList->at(1)->expr->name, "t5");
 | ||
| 
 | ||
|   ASSERT_TRUE(stmt->selectList->at(0)->windowDescription->frameDescription);
 | ||
|   ASSERT_EQ(stmt->selectList->at(0)->windowDescription->frameDescription->type, kRows);
 | ||
|   ASSERT_TRUE(stmt->selectList->at(0)->windowDescription->frameDescription->start);
 | ||
|   ASSERT_EQ(stmt->selectList->at(0)->windowDescription->frameDescription->start->offset, 0);
 | ||
|   ASSERT_EQ(stmt->selectList->at(0)->windowDescription->frameDescription->start->type, kPreceding);
 | ||
|   ASSERT_TRUE(stmt->selectList->at(0)->windowDescription->frameDescription->start->unbounded);
 | ||
|   ASSERT_TRUE(stmt->selectList->at(0)->windowDescription->frameDescription->end);
 | ||
|   ASSERT_EQ(stmt->selectList->at(0)->windowDescription->frameDescription->end->offset, 0);
 | ||
|   ASSERT_EQ(stmt->selectList->at(0)->windowDescription->frameDescription->end->type, kCurrentRow);
 | ||
|   ASSERT_FALSE(stmt->selectList->at(0)->windowDescription->frameDescription->end->unbounded);
 | ||
| 
 | ||
|   const auto frame_starts =
 | ||
|       std::vector<FrameBound>{{25, kPreceding, false}, {0, kPreceding, true}, {0, kPreceding, true}};
 | ||
|   const auto frame_ends =
 | ||
|       std::vector<FrameBound>{{2, kFollowing, false}, {0, kFollowing, true}, {0, kCurrentRow, false}};
 | ||
| 
 | ||
|   for (auto bound_index = size_t{0}; bound_index < frame_starts.size(); ++bound_index) {
 | ||
|     stmt = (SelectStatement*)result.getStatement(2 + bound_index);
 | ||
|     const auto& expected_start = frame_starts[bound_index];
 | ||
|     const auto& expected_end = frame_ends[bound_index];
 | ||
| 
 | ||
|     ASSERT_TRUE(stmt->selectList);
 | ||
|     ASSERT_EQ(stmt->selectList->size(), 1);
 | ||
|     ASSERT_TRUE(stmt->fromTable);
 | ||
|     ASSERT_EQ(stmt->fromTable->type, kTableName);
 | ||
|     ASSERT_STREQ(stmt->fromTable->name, "t");
 | ||
| 
 | ||
|     ASSERT_EQ(stmt->selectList->at(0)->type, kExprFunctionRef);
 | ||
|     ASSERT_STREQ(stmt->selectList->at(0)->name, "rank");
 | ||
|     ASSERT_TRUE(stmt->selectList->at(0)->exprList->empty());
 | ||
| 
 | ||
|     ASSERT_TRUE(stmt->selectList->at(0)->windowDescription);
 | ||
|     ASSERT_TRUE(stmt->selectList->at(0)->windowDescription->partitionList);
 | ||
|     ASSERT_EQ(stmt->selectList->at(0)->windowDescription->partitionList->size(), 1);
 | ||
|     ASSERT_EQ(stmt->selectList->at(0)->windowDescription->partitionList->at(0)->type, kExprColumnRef);
 | ||
|     ASSERT_STREQ(stmt->selectList->at(0)->windowDescription->partitionList->at(0)->name, "t1");
 | ||
| 
 | ||
|     ASSERT_TRUE(stmt->selectList->at(0)->windowDescription->orderList);
 | ||
|     ASSERT_EQ(stmt->selectList->at(0)->windowDescription->orderList->size(), 1);
 | ||
|     ASSERT_EQ(stmt->selectList->at(0)->windowDescription->orderList->at(0)->type, kOrderAsc);
 | ||
|     ASSERT_TRUE(stmt->selectList->at(0)->windowDescription->orderList->at(0)->expr);
 | ||
|     ASSERT_EQ(stmt->selectList->at(0)->windowDescription->orderList->at(0)->expr->type, kExprColumnRef);
 | ||
|     ASSERT_STREQ(stmt->selectList->at(0)->windowDescription->orderList->at(0)->expr->name, "t2");
 | ||
|     ASSERT_TRUE(stmt->selectList->at(0)->windowDescription->frameDescription);
 | ||
|     ASSERT_TRUE(stmt->selectList->at(0)->windowDescription->frameDescription->start);
 | ||
|     ASSERT_EQ(stmt->selectList->at(0)->windowDescription->frameDescription->start->offset, expected_start.offset);
 | ||
|     ASSERT_EQ(stmt->selectList->at(0)->windowDescription->frameDescription->start->type, expected_start.type);
 | ||
|     ASSERT_EQ(stmt->selectList->at(0)->windowDescription->frameDescription->start->unbounded, expected_start.unbounded);
 | ||
|     ASSERT_TRUE(stmt->selectList->at(0)->windowDescription->frameDescription->end);
 | ||
|     ASSERT_EQ(stmt->selectList->at(0)->windowDescription->frameDescription->end->offset, expected_end.offset);
 | ||
|     ASSERT_EQ(stmt->selectList->at(0)->windowDescription->frameDescription->end->type, expected_end.type);
 | ||
|     ASSERT_EQ(stmt->selectList->at(0)->windowDescription->frameDescription->end->unbounded, expected_end.unbounded);
 | ||
|   }
 | ||
| }
 | 
