mirror of
https://github.com/MariaDB/server.git
synced 2026-01-13 06:03:12 +00:00
380 lines
11 KiB
Objective-C
380 lines
11 KiB
Objective-C
/* Copyright (c) 2009, 2010, Oracle and/or its affiliates. All rights reserved.
|
|
|
|
This program is free software; you can redistribute it and/or modify
|
|
it under the terms of the GNU General Public License as published by
|
|
the Free Software Foundation; version 2 of the License.
|
|
|
|
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 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-1335 USA */
|
|
|
|
/**
|
|
@file Representation of an SQL command.
|
|
*/
|
|
|
|
#ifndef SQL_CMD_INCLUDED
|
|
#define SQL_CMD_INCLUDED
|
|
|
|
#include <my_base.h>
|
|
|
|
#include "sql_command.h"
|
|
|
|
class Storage_engine_name
|
|
{
|
|
protected:
|
|
LEX_CSTRING m_storage_engine_name;
|
|
public:
|
|
Storage_engine_name()
|
|
{
|
|
m_storage_engine_name.str= NULL;
|
|
m_storage_engine_name.length= 0;
|
|
}
|
|
Storage_engine_name(const LEX_CSTRING &name)
|
|
:m_storage_engine_name(name)
|
|
{ }
|
|
bool resolve_storage_engine_with_error(THD *thd,
|
|
handlerton **ha,
|
|
bool tmp_table);
|
|
bool is_set() { return m_storage_engine_name.str != NULL; }
|
|
const LEX_CSTRING *name() const { return &m_storage_engine_name; }
|
|
};
|
|
|
|
|
|
class Prepared_statement;
|
|
|
|
/**
|
|
@class Sql_cmd - Representation of an SQL command.
|
|
|
|
This class is an interface between the parser and the runtime.
|
|
The parser builds the appropriate derived classes of Sql_cmd
|
|
to represent a SQL statement in the parsed tree.
|
|
The execute() method in the derived classes of Sql_cmd contain the runtime
|
|
implementation.
|
|
Note that this interface is used for SQL statements recently implemented,
|
|
the code for older statements tend to load the LEX structure with more
|
|
attributes instead.
|
|
Implement new statements by sub-classing Sql_cmd, as this improves
|
|
code modularity (see the 'big switch' in dispatch_command()), and decreases
|
|
the total size of the LEX structure (therefore saving memory in stored
|
|
programs).
|
|
The recommended name of a derived class of Sql_cmd is Sql_cmd_<derived>.
|
|
|
|
Notice that the Sql_cmd class should not be confused with the
|
|
Statement class. Statement is a class that is used to manage an SQL
|
|
command or a set of SQL commands. When the SQL statement text is
|
|
analyzed, the parser will create one or more Sql_cmd objects to
|
|
represent the actual SQL commands.
|
|
*/
|
|
class Sql_cmd : public Sql_alloc
|
|
{
|
|
private:
|
|
Sql_cmd(const Sql_cmd &); // No copy constructor wanted
|
|
void operator=(Sql_cmd &); // No assignment operator wanted
|
|
|
|
public:
|
|
/**
|
|
@brief Return the command code for this statement
|
|
*/
|
|
virtual enum_sql_command sql_command_code() const = 0;
|
|
|
|
/**
|
|
@brief Check whether the statement has been prepared
|
|
@returns true if this statement is prepared, false otherwise
|
|
*/
|
|
bool is_prepared() const { return m_prepared; }
|
|
|
|
/**
|
|
@brief Prepare this SQL statement
|
|
@param thd global context the processed statement
|
|
@returns false if success, true if error
|
|
*/
|
|
virtual bool prepare(THD *thd)
|
|
{
|
|
/* Default behavior for a statement is to have no preparation code. */
|
|
DBUG_ASSERT(!is_prepared());
|
|
set_prepared();
|
|
return false;
|
|
}
|
|
|
|
/**
|
|
@brief Execute this SQL statement
|
|
@param thd global context the processed statement
|
|
@returns false if success, true if error
|
|
*/
|
|
virtual bool execute(THD *thd) = 0;
|
|
|
|
virtual Storage_engine_name *option_storage_engine_name()
|
|
{
|
|
return NULL;
|
|
}
|
|
|
|
/**
|
|
@brief Set the owning prepared statement
|
|
*/
|
|
void set_owner(Prepared_statement *stmt) { m_owner = stmt; }
|
|
|
|
/**
|
|
@breaf Get the owning prepared statement
|
|
*/
|
|
Prepared_statement *get_owner() { return m_owner; }
|
|
|
|
/**
|
|
@brief Check whether this command is a DML statement
|
|
@return true if SQL command is a DML statement, false otherwise
|
|
*/
|
|
virtual bool is_dml() const { return false; }
|
|
|
|
virtual void get_dml_stat (ha_rows &found, ha_rows &changed)
|
|
{
|
|
found= changed= 0;
|
|
}
|
|
|
|
/**
|
|
@brief Unprepare prepared statement for the command
|
|
@param thd global context of the processed statement
|
|
|
|
@notes
|
|
Temporary function used to "unprepare" a prepared statement after
|
|
preparation, so that a subsequent execute statement will reprepare it.
|
|
This is done because UNIT::cleanup() will un-resolve all resolved QBs.
|
|
*/
|
|
virtual void unprepare(THD *thd)
|
|
{
|
|
DBUG_ASSERT(is_prepared());
|
|
m_prepared = false;
|
|
}
|
|
|
|
protected:
|
|
Sql_cmd() : m_prepared(false), m_owner(nullptr)
|
|
{}
|
|
|
|
virtual ~Sql_cmd()
|
|
{
|
|
/*
|
|
Sql_cmd objects are allocated in thd->mem_root.
|
|
In MySQL, the C++ destructor is never called, the underlying MEM_ROOT is
|
|
simply destroyed instead.
|
|
Do not rely on the destructor for any cleanup.
|
|
*/
|
|
DBUG_ASSERT(false);
|
|
}
|
|
|
|
/**
|
|
@brief Set this statement as prepared
|
|
*/
|
|
void set_prepared() { m_prepared = true; }
|
|
|
|
private:
|
|
/* True when statement has been prepared */
|
|
bool m_prepared;
|
|
/* Owning prepared statement, nullptr if not prepared */
|
|
Prepared_statement *m_owner;
|
|
|
|
};
|
|
|
|
struct LEX;
|
|
class select_result;
|
|
class Prelocking_strategy;
|
|
class DML_prelocking_strategy;
|
|
class Protocol;
|
|
|
|
/**
|
|
@class Sql_cmd_dml - derivative abstract class used for DML statements
|
|
|
|
This class is a class derived from Sql_cmd used when processing such
|
|
data manipulation commands as SELECT, INSERT, UPDATE, DELETE and others
|
|
that operate over some tables.
|
|
After the parser phase all these commands are supposed to be processed
|
|
by the same schema:
|
|
- precheck of the access rights is performed for the used tables
|
|
- the used tables are opened
|
|
- context analysis phase is performed for the statement
|
|
- the used tables are locked
|
|
- the statement is optimized and executed
|
|
- clean-up is performed for the statement.
|
|
This schema is reflected in the function Sql_cmd_dml::execute() that
|
|
uses Sql_cmd_dml::prepare is the statement has not been prepared yet.
|
|
Precheck of the access right, context analysis are specific for statements
|
|
of a certain type. That's why the methods implementing this operations are
|
|
declared as abstract in this class.
|
|
|
|
@note
|
|
Currently this class is used only for UPDATE and DELETE commands.
|
|
*/
|
|
class Sql_cmd_dml : public Sql_cmd
|
|
{
|
|
public:
|
|
|
|
/**
|
|
@brief Check whether the statement changes the contents of used tables
|
|
@return true if this is data change statement, false otherwise
|
|
*/
|
|
virtual bool is_data_change_stmt() const { return true; }
|
|
|
|
/**
|
|
@brief Perform context analysis of the statement
|
|
@param thd global context the processed statement
|
|
@returns false on success, true on error
|
|
*/
|
|
bool prepare(THD *thd) override;
|
|
|
|
/**
|
|
Execute the processed statement once
|
|
@param thd global context the processed statement
|
|
@returns false on success, true on error
|
|
*/
|
|
bool execute(THD *thd) override;
|
|
|
|
bool is_dml() const override { return true; }
|
|
|
|
select_result *get_result() { return result; }
|
|
|
|
protected:
|
|
Sql_cmd_dml()
|
|
: Sql_cmd(), lex(nullptr), result(nullptr),
|
|
m_empty_query(false)
|
|
{}
|
|
|
|
/**
|
|
@brief Check whether query is guaranteed to return no data
|
|
@return true if query is guaranteed to return no data, false otherwise
|
|
|
|
@todo Also check this for the following cases:
|
|
- Empty source for multi-table UPDATE and DELETE.
|
|
- Check empty query expression for INSERT
|
|
*/
|
|
bool is_empty_query() const
|
|
{
|
|
DBUG_ASSERT(is_prepared());
|
|
return m_empty_query;
|
|
}
|
|
|
|
/**
|
|
@brief Set statement as returning no data
|
|
*/
|
|
void set_empty_query() { m_empty_query = true; }
|
|
|
|
/**
|
|
@brief Perform precheck of table privileges for the specific command
|
|
@param thd global context the processed statement
|
|
@returns false if success, true if false
|
|
|
|
@details
|
|
Check that user has some relevant privileges for all tables involved in
|
|
the statement, e.g. SELECT privileges for tables selected from, INSERT
|
|
privileges for tables inserted into, etc. This function will also populate
|
|
TABLE_LIST::grant with all privileges the user has for each table, which
|
|
is later used during checking of column privileges.
|
|
Note that at preparation time, views are not expanded yet. Privilege
|
|
checking is thus rudimentary and must be complemented with later calls to
|
|
SELECT_LEX::check_view_privileges().
|
|
The reason to call this function at such an early stage is to be able to
|
|
quickly reject statements for which the user obviously has insufficient
|
|
privileges.
|
|
*/
|
|
virtual bool precheck(THD *thd) = 0;
|
|
|
|
/**
|
|
@brief Perform the command-specific actions of the context analysis
|
|
@param thd global context the processed statement
|
|
@returns false if success, true if error
|
|
|
|
@note
|
|
This function is called from prepare()
|
|
*/
|
|
virtual bool prepare_inner(THD *thd) = 0;
|
|
|
|
/**
|
|
@brief Perform the command-specific actions of optimization and excution
|
|
@param thd global context the processed statement
|
|
@returns false on success, true on error
|
|
*/
|
|
virtual bool execute_inner(THD *thd);
|
|
|
|
virtual DML_prelocking_strategy *get_dml_prelocking_strategy() = 0;
|
|
|
|
uint table_count;
|
|
|
|
protected:
|
|
LEX *lex; /**< Pointer to LEX for this statement */
|
|
select_result *result; /**< Pointer to object for handling of the result */
|
|
bool m_empty_query; /**< True if query will produce no rows */
|
|
};
|
|
|
|
|
|
class Sql_cmd_show_slave_status: public Sql_cmd
|
|
{
|
|
protected:
|
|
bool show_all_slaves_status;
|
|
public:
|
|
Sql_cmd_show_slave_status()
|
|
:show_all_slaves_status(false)
|
|
{}
|
|
|
|
Sql_cmd_show_slave_status(bool status_all)
|
|
:show_all_slaves_status(status_all)
|
|
{}
|
|
|
|
enum_sql_command sql_command_code() const override { return SQLCOM_SHOW_SLAVE_STAT; }
|
|
|
|
bool execute(THD *thd) override;
|
|
bool is_show_all_slaves_stat() { return show_all_slaves_status; }
|
|
};
|
|
|
|
|
|
class Sql_cmd_create_table_like: public Sql_cmd,
|
|
public Storage_engine_name
|
|
{
|
|
public:
|
|
Storage_engine_name *option_storage_engine_name() override { return this; }
|
|
bool execute(THD *thd) override;
|
|
};
|
|
|
|
class Sql_cmd_create_table: public Sql_cmd_create_table_like
|
|
{
|
|
public:
|
|
enum_sql_command sql_command_code() const override { return SQLCOM_CREATE_TABLE; }
|
|
};
|
|
|
|
class Sql_cmd_create_sequence: public Sql_cmd_create_table_like
|
|
{
|
|
public:
|
|
enum_sql_command sql_command_code() const override { return SQLCOM_CREATE_SEQUENCE; }
|
|
};
|
|
|
|
|
|
/**
|
|
Sql_cmd_call represents the CALL statement.
|
|
*/
|
|
class Sql_cmd_call : public Sql_cmd
|
|
{
|
|
public:
|
|
class sp_name *m_name;
|
|
const class Sp_handler *m_handler;
|
|
Sql_cmd_call(class sp_name *name, const class Sp_handler *handler)
|
|
:m_name(name),
|
|
m_handler(handler)
|
|
{}
|
|
|
|
virtual ~Sql_cmd_call() = default;
|
|
|
|
/**
|
|
Execute a CALL statement at runtime.
|
|
@param thd the current thread.
|
|
@return false on success.
|
|
*/
|
|
bool execute(THD *thd) override;
|
|
|
|
enum_sql_command sql_command_code() const override
|
|
{
|
|
return SQLCOM_CALL;
|
|
}
|
|
};
|
|
|
|
#endif // SQL_CMD_INCLUDED
|