unittest/shell_sql_t.cc (370 lines of code) (raw):
/* Copyright (c) 2014, 2024, Oracle and/or its affiliates.
This program is free software; you can redistribute it and/or modify
it under the terms of the GNU General Public License, version 2.0,
as published by the Free Software Foundation.
This program is designed to work with certain software (including
but not limited to OpenSSL) that is licensed under separate terms,
as designated in a particular file or component or in included license
documentation. The authors of MySQL hereby grant you an additional
permission to link the program and your derivative works with the
separately licensed software that they have either included with
the program or referenced in the documentation.
This program is distributed in the hope that it will be useful, but
WITHOUT ANY WARRANTY; without even the implied warranty of
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See
the GNU General Public License, version 2.0, for more details.
You should have received a copy of the GNU General Public License
along with this program; if not, write to the Free Software Foundation, Inc.,
51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA */
#include <cstdio>
#include <cstdlib>
#include <fstream>
#include <string>
#include "gtest_clean.h"
#include "scripting/lang_base.h"
#include "scripting/types.h"
#include "scripting/types_cpp.h"
#include "modules/mod_mysql_session.h"
#include "modules/mod_shell.h"
#include "mysqlshdk/libs/utils/utils_general.h"
#include "mysqlshdk/shellcore/shell_console.h"
#include "scripting/common.h"
#include "shellcore/base_session.h"
#include "shellcore/shell_core.h"
#include "shellcore/shell_sql.h"
#include "unittest/test_utils.h"
using namespace std::placeholders;
namespace shcore {
namespace sql_shell_tests {
class Environment {
public:
Environment() : m_console(&output_handler.deleg) {
shell_core = std::make_shared<Shell_core>();
shell_sql = std::make_shared<Shell_sql>(shell_core.get());
}
~Environment() {}
Shell_test_output_handler output_handler;
mysqlsh::Shell_console m_console;
std::shared_ptr<Shell_core> shell_core;
std::shared_ptr<Shell_sql> shell_sql;
};
class Shell_sql_test : public Shell_core_test_wrapper {
protected:
Environment env;
virtual void SetUp() {
Shell_core_test_wrapper::SetUp();
connect();
}
virtual void TearDown() {
shcore::Argument_list args;
auto session = env.shell_core->get_dev_session();
if (session) {
session->close();
}
}
void process_sql_result(const std::shared_ptr<mysqlshdk::db::IResult> &,
const shcore::Sql_result_info &) {}
void handle_input(std::string &query, Input_state &state) {
env.shell_sql->set_result_processor(
std::bind(&Shell_sql_test::process_sql_result, this, _1, _2));
std::string q = query + "\n";
env.shell_sql->handle_input(q, state);
query = q;
}
void connect() {
const char *pwd = getenv("MYSQL_PWD");
const char *port = getenv("MYSQL_PORT");
std::string mysql_uri = "mysql://root@localhost";
mysql_uri.append(":");
mysql_uri.append(port);
auto connection_options = shcore::get_connection_options(mysql_uri);
if (pwd)
connection_options.set_password(pwd);
else
connection_options.set_password("");
auto session = std::make_shared<mysqlsh::mysql::ClassicSession>();
session->connect(connection_options);
env.shell_core->set_dev_session(session);
}
};
TEST_F(Shell_sql_test, test_initial_state) {
EXPECT_EQ("", env.shell_sql->get_continued_input_context());
}
TEST_F(Shell_sql_test, full_statement) {
Input_state state = Input_state::Ok;
std::string query = "show databases;";
handle_input(query, state);
// Query totally executed, input should be OK
// Handled contains the executed statement
// Prompt is for full statement
EXPECT_EQ(Input_state::Ok, state);
EXPECT_EQ("", query);
EXPECT_EQ("show databases;", env.shell_sql->get_handled_input());
EXPECT_EQ("", env.shell_sql->get_continued_input_context());
}
TEST_F(Shell_sql_test, sql_multi_line_statement) {
Input_state state = Input_state::Ok;
std::string query = "show";
handle_input(query, state);
// Nothing is executed until the delimiter is reached and the prompt changes
// Prompt changes to multiline mode
EXPECT_EQ(Input_state::ContinuedSingle, state);
// EXPECT_EQ("show", query);
EXPECT_EQ("", env.shell_sql->get_handled_input());
EXPECT_EQ("-", env.shell_sql->get_continued_input_context());
// Caching the partial statements is now internal
// we just send whatever is remaining for the query to execute
query.append("databases");
handle_input(query, state);
// Nothing is executed until the delimiter is reached and the prompt changes
// Prompt changes to multiline
EXPECT_EQ(Input_state::ContinuedSingle, state);
// EXPECT_EQ("databases", query);
EXPECT_EQ("", env.shell_sql->get_handled_input());
EXPECT_EQ("-", env.shell_sql->get_continued_input_context());
query.append(";");
handle_input(query, state);
// Nothing is executed until the delimiter is reached and the prompt changes
// Prompt changes to multiline
EXPECT_EQ(Input_state::Ok, state);
EXPECT_EQ("", query);
EXPECT_EQ("show\ndatabases\n;", env.shell_sql->get_handled_input());
EXPECT_EQ("", env.shell_sql->get_continued_input_context());
}
TEST_F(Shell_sql_test, sql_multi_line_string_delimiter) {
Input_state state = Input_state::Ok;
std::string query = "delimiter %%%";
handle_input(query, state);
// Nothing is executed until the delimiter is reached and the prompt changes
// Prompt changes to multiline mode
EXPECT_EQ(Input_state::Ok, state);
query.append("show");
handle_input(query, state);
// Nothing is executed until the delimiter is reached and the prompt changes
// Prompt changes to multiline mode
EXPECT_EQ(Input_state::ContinuedSingle, state);
// EXPECT_EQ("show", query);
EXPECT_EQ("", env.shell_sql->get_handled_input());
EXPECT_EQ("-", env.shell_sql->get_continued_input_context());
// Caching the partial statements is now internal
// we just send whatever is remaining for the query to execute
query.append("databases");
handle_input(query, state);
// Nothing is executed until the delimiter is reached and the prompt changes
// Prompt changes to multiline
EXPECT_EQ(Input_state::ContinuedSingle, state);
// EXPECT_EQ("databases", query);
EXPECT_EQ("", env.shell_sql->get_handled_input());
EXPECT_EQ("-", env.shell_sql->get_continued_input_context());
query.append("%%%");
handle_input(query, state);
// Nothing is executed until the delimiter is reached and the prompt changes
// Prompt changes to multiline
EXPECT_EQ(Input_state::Ok, state);
EXPECT_EQ("", query);
EXPECT_EQ("show\ndatabases\n%%%", env.shell_sql->get_handled_input());
EXPECT_EQ("", env.shell_sql->get_continued_input_context());
query.append("delimiter ;");
handle_input(query, state);
// Nothing is executed until the delimiter is reached and the prompt changes
// Prompt changes to multiline mode
EXPECT_EQ(Input_state::Ok, state);
}
TEST_F(Shell_sql_test, multiple_statements_continued) {
Input_state state = Input_state::Ok;
std::string query = "show databases; show";
handle_input(query, state);
// The first statement will be handled but the second will be hold
// until the delimiter is found.
// Prompt changes to multiline
// query will be updated to only keep what has not been executed
EXPECT_EQ(Input_state::ContinuedSingle, state);
// EXPECT_EQ("show", query);
EXPECT_EQ("show databases;", env.shell_sql->get_handled_input());
EXPECT_EQ("-", env.shell_sql->get_continued_input_context());
query.append("databases;");
handle_input(query, state);
EXPECT_EQ(Input_state::Ok, state);
EXPECT_EQ("", query);
EXPECT_EQ("show\ndatabases;", env.shell_sql->get_handled_input());
EXPECT_EQ("", env.shell_sql->get_continued_input_context());
}
TEST_F(Shell_sql_test, global_multi_line_statement_ignored) {
Input_state state = Input_state::Ok;
std::string query = "show";
handle_input(query, state);
// Nothing is executed until the delimiter is reached and the prompt changes
// Prompt changes to multiline mode
EXPECT_EQ(Input_state::ContinuedSingle, state);
EXPECT_EQ("", env.shell_sql->get_handled_input());
EXPECT_EQ("-", env.shell_sql->get_continued_input_context());
query.append("databases;\nshow");
handle_input(query, state);
// Being global multiline make sthe statement to be executed right away
EXPECT_EQ(Input_state::ContinuedSingle, state);
EXPECT_EQ("show\ndatabases;", env.shell_sql->get_handled_input());
EXPECT_EQ("-", env.shell_sql->get_continued_input_context());
query.append("databases;");
handle_input(query, state);
// Being global multiline make sthe statement to be executed right away
EXPECT_EQ(Input_state::Ok, state);
EXPECT_EQ("show\ndatabases;", env.shell_sql->get_handled_input());
EXPECT_EQ("", env.shell_sql->get_continued_input_context());
}
TEST_F(Shell_sql_test, multiple_statements_and_continued) {
Input_state state = Input_state::Ok;
std::string query = "show databases; select 1;show";
handle_input(query, state);
// The first statement will be handled but the second will be hold
// until the delimiter is found.
// Prompt changes to multiline
// query will be updated to only keep what has not been executed
EXPECT_EQ(Input_state::ContinuedSingle, state);
// EXPECT_EQ("show", query);
EXPECT_EQ("show databases;select 1;", env.shell_sql->get_handled_input());
EXPECT_EQ("-", env.shell_sql->get_continued_input_context());
query.append("databases;");
handle_input(query, state);
EXPECT_EQ(Input_state::Ok, state);
EXPECT_EQ("", query);
EXPECT_EQ("show\ndatabases;", env.shell_sql->get_handled_input());
EXPECT_EQ("", env.shell_sql->get_continued_input_context());
}
TEST_F(Shell_sql_test, multiline_comment) {
Input_state state = Input_state::Ok;
std::string query = "/*";
handle_input(query, state);
EXPECT_EQ(Input_state::ContinuedSingle, state);
EXPECT_EQ("/*\n", query);
EXPECT_EQ("", env.shell_sql->get_handled_input());
EXPECT_EQ("/*", env.shell_sql->get_continued_input_context());
query.append("this was a multiline comment");
handle_input(query, state);
EXPECT_EQ(Input_state::ContinuedSingle, state);
EXPECT_EQ("/*\nthis was a multiline comment\n", query);
EXPECT_EQ("", env.shell_sql->get_handled_input());
EXPECT_EQ("/*", env.shell_sql->get_continued_input_context());
query.append("*/;");
handle_input(query, state);
EXPECT_EQ(Input_state::Ok, state);
EXPECT_EQ("", query);
EXPECT_EQ("/*\nthis was a multiline comment\n*/;",
env.shell_sql->get_handled_input());
EXPECT_EQ("", env.shell_sql->get_continued_input_context());
}
TEST_F(Shell_sql_test, multiline_single_quote_continued_string) {
Input_state state = Input_state::Ok;
std::string query = "select 'hello ";
handle_input(query, state);
EXPECT_EQ(Input_state::ContinuedSingle, state);
// EXPECT_EQ("select 'hello ", query);
EXPECT_EQ("", env.shell_sql->get_handled_input());
EXPECT_EQ("'", env.shell_sql->get_continued_input_context());
query.append("world';");
handle_input(query, state);
EXPECT_EQ(Input_state::Ok, state);
EXPECT_EQ("", query);
EXPECT_EQ("select 'hello \nworld';", env.shell_sql->get_handled_input());
EXPECT_EQ("", env.shell_sql->get_continued_input_context());
}
TEST_F(Shell_sql_test, multiline_double_quote_continued_string) {
Input_state state = Input_state::Ok;
std::string query = "select \"hello ";
handle_input(query, state);
EXPECT_EQ(Input_state::ContinuedSingle, state);
// EXPECT_EQ("select \"hello ", query);
EXPECT_EQ("", env.shell_sql->get_handled_input());
EXPECT_EQ("\"", env.shell_sql->get_continued_input_context());
query.append("world\";");
handle_input(query, state);
EXPECT_EQ(Input_state::Ok, state);
EXPECT_EQ("", query);
EXPECT_EQ("select \"hello \nworld\";", env.shell_sql->get_handled_input());
EXPECT_EQ("", env.shell_sql->get_continued_input_context());
}
TEST_F(Shell_sql_test, DISABLED_multiline_backtick_string) {
Input_state state = Input_state::Ok;
std::string query = "select * from `t";
handle_input(query, state);
EXPECT_EQ(Input_state::ContinuedSingle, state);
EXPECT_EQ("select * from `sakila`.`", query);
EXPECT_EQ("", env.shell_sql->get_handled_input());
EXPECT_EQ("`", env.shell_sql->get_continued_input_context());
query = "film`;";
handle_input(query, state);
EXPECT_EQ(Input_state::Ok, state);
EXPECT_EQ("", query);
EXPECT_EQ("", env.shell_sql->get_handled_input());
EXPECT_EQ("", env.shell_sql->get_continued_input_context());
}
TEST_F(Shell_sql_test, multiple_single_double_quotes) {
Input_state state = Input_state::Ok;
std::string query = "SELECT '''' as a;";
handle_input(query, state);
EXPECT_EQ(Input_state::Ok, state);
EXPECT_EQ("", query);
EXPECT_EQ("SELECT '''' as a;", env.shell_sql->get_handled_input());
EXPECT_EQ("", env.shell_sql->get_continued_input_context());
query = "SELECT \"\"\"\" as a;";
handle_input(query, state);
EXPECT_EQ(Input_state::Ok, state);
EXPECT_EQ("", query);
EXPECT_EQ("SELECT \"\"\"\" as a;", env.shell_sql->get_handled_input());
EXPECT_EQ("", env.shell_sql->get_continued_input_context());
query = "SELECT \"\'\" as a;";
handle_input(query, state);
EXPECT_EQ(Input_state::Ok, state);
EXPECT_EQ("", query);
EXPECT_EQ("SELECT \"\'\" as a;", env.shell_sql->get_handled_input());
EXPECT_EQ("", env.shell_sql->get_continued_input_context());
query = "SELECT '\\'' as a;";
handle_input(query, state);
EXPECT_EQ(Input_state::Ok, state);
EXPECT_EQ("", query);
EXPECT_EQ("SELECT '\\'' as a;", env.shell_sql->get_handled_input());
EXPECT_EQ("", env.shell_sql->get_continued_input_context());
}
TEST_F(Shell_sql_test, stmt_inline_comment) {
// A multiline comment in the middle of a statement is
// not trimmed off
Input_state state = Input_state::Ok;
std::string query = "SELECT 1 /* This is an inline comment */AS _one;";
handle_input(query, state);
EXPECT_EQ(Input_state::Ok, state);
EXPECT_EQ("", query);
// SQL has not been handled yet
EXPECT_EQ("SELECT 1 /* This is an inline comment */AS _one;",
env.shell_sql->get_handled_input());
}
TEST_F(Shell_sql_test, continued_stmt_multiline_comment) {
// A multiline comment in the middle of a statement is
// not trimmed off
Input_state state = Input_state::Ok;
std::string query = "SELECT 1 AS _one /*";
handle_input(query, state);
EXPECT_EQ(Input_state::ContinuedSingle, state);
EXPECT_EQ("SELECT 1 AS _one /*\n", query);
// SQL has not been handled yet
EXPECT_EQ("", env.shell_sql->get_handled_input());
// Prompt for continued statement
EXPECT_EQ("/*", env.shell_sql->get_continued_input_context());
query.append("comment text */");
handle_input(query, state);
EXPECT_EQ(Input_state::ContinuedSingle, state);
EXPECT_EQ("SELECT 1 AS _one /*\ncomment text */\n", query);
// SQL has not been handled yet
EXPECT_EQ("", env.shell_sql->get_handled_input());
// Prompt for continued statement - context is a comment because parsing is
// stopped until delimiter comes;
EXPECT_EQ("/*", env.shell_sql->get_continued_input_context());
query.append(";");
handle_input(query, state);
EXPECT_EQ(Input_state::Ok, state);
EXPECT_EQ("", query);
EXPECT_EQ("SELECT 1 AS _one /*\ncomment text */\n;",
env.shell_sql->get_handled_input());
EXPECT_EQ("", env.shell_sql->get_continued_input_context());
}
TEST_F(Shell_sql_test, continued_stmt_dash_dash_comment) {
// ATM dash dash comments are trimmed from statements
Input_state state = Input_state::Ok;
std::string query = "select 1 as one -- sample comment";
handle_input(query, state);
EXPECT_EQ(Input_state::ContinuedSingle, state);
EXPECT_EQ("select 1 as one -- sample comment\n", query);
// SQL has not been handled yet
EXPECT_EQ("", env.shell_sql->get_handled_input());
// Prompt for continued statement
EXPECT_EQ("-", env.shell_sql->get_continued_input_context());
query.append(";select 2 as two;");
handle_input(query, state);
EXPECT_EQ(Input_state::Ok, state);
EXPECT_EQ("", query);
EXPECT_EQ("select 1 as one -- sample comment\n;select 2 as two;",
env.shell_sql->get_handled_input());
EXPECT_EQ("", env.shell_sql->get_continued_input_context());
}
TEST_F(Shell_sql_test, continued_stmt_dash_dash_comment_batch) {
// ATM dash dash comments are trimmed from statements
Input_state state = Input_state::Ok;
std::string query = "select 1 as one -- sample comment\n;select 2 as two;";
handle_input(query, state);
EXPECT_EQ(Input_state::Ok, state);
EXPECT_EQ("", query);
EXPECT_EQ("select 1 as one -- sample comment\n;select 2 as two;",
env.shell_sql->get_handled_input());
EXPECT_EQ("", env.shell_sql->get_continued_input_context());
}
TEST_F(Shell_sql_test, continued_stmt_hash_comment) {
// ATM hash comments are trimmed from statements
Input_state state = Input_state::Ok;
std::string query = "select 1 as one #sample comment";
handle_input(query, state);
EXPECT_EQ(Input_state::ContinuedSingle, state);
EXPECT_EQ("select 1 as one #sample comment\n", query);
// SQL has not been handled yet
EXPECT_EQ("", env.shell_sql->get_handled_input());
// Prompt for continued statement
EXPECT_EQ("-", env.shell_sql->get_continued_input_context());
query.append(";select 2 as two;");
handle_input(query, state);
EXPECT_EQ(Input_state::Ok, state);
EXPECT_EQ("", query);
EXPECT_EQ("select 1 as one #sample comment\n;select 2 as two;",
env.shell_sql->get_handled_input());
EXPECT_EQ("", env.shell_sql->get_continued_input_context());
}
TEST_F(Shell_sql_test, continued_stmt_hash_comment_batch) {
// ATM hash comments are trimmed from statements
Input_state state = Input_state::Ok;
std::string query = "select 1 as one #sample comment\n;select 2 as two;";
handle_input(query, state);
EXPECT_EQ(Input_state::Ok, state);
EXPECT_EQ("", query);
EXPECT_EQ("select 1 as one #sample comment\n;select 2 as two;",
env.shell_sql->get_handled_input());
EXPECT_EQ("", env.shell_sql->get_continued_input_context());
}
TEST_F(Shell_sql_test, batch_script) {
Input_state state = Input_state::Ok;
std::string query = "select 1;\nselect 2;\nselect 3;\n";
handle_input(query, state);
EXPECT_EQ(Input_state::Ok, state);
EXPECT_EQ("", query);
EXPECT_EQ("select 1;select 2;select 3;", env.shell_sql->get_handled_input());
EXPECT_EQ("", env.output_handler.std_out);
}
TEST_F(Shell_sql_test, batch_script_error) {
Input_state state = Input_state::Ok;
std::string query = "select 1;\ndrop schema badschema_;\nselect 3;\n";
handle_input(query, state);
EXPECT_EQ(Input_state::Ok, state);
EXPECT_EQ("", query);
EXPECT_EQ("select 1;drop schema badschema_;select 3;",
env.shell_sql->get_handled_input());
EXPECT_EQ("", env.output_handler.std_out);
}
TEST_F(Shell_sql_test, batch_script_error_force) {}
} // namespace sql_shell_tests
} // namespace shcore