mirror of
https://github.com/MariaDB/server.git
synced 2025-01-25 00:04:33 +01:00
36290c0923
This patch also fixes Bug#55452 "SET PASSWORD is replicated twice in RBR mode". The goal of this patch is to remove the release of metadata locks from close_thread_tables(). This is necessary to not mistakenly release the locks in the course of a multi-step operation that involves multiple close_thread_tables() or close_tables_for_reopen(). On the same token, move statement commit outside close_thread_tables(). Other cleanups: Cleanup COM_FIELD_LIST. Don't call close_thread_tables() in COM_SHUTDOWN -- there are no open tables there that can be closed (we leave the locked tables mode in THD destructor, and this close_thread_tables() won't leave it anyway). Make open_and_lock_tables() and open_and_lock_tables_derived() call close_thread_tables() upon failure. Remove the calls to close_thread_tables() that are now unnecessary. Simplify the back off condition in Open_table_context. Streamline metadata lock handling in LOCK TABLES implementation. Add asserts to ensure correct life cycle of statement transaction in a session. Remove a piece of dead code that has also become redundant after the fix for Bug 37521. mysql-test/r/variables.result: Update results: set @@autocommit and statement transaction/ prelocked mode. mysql-test/r/view.result: A harmless change in CHECK TABLE <view> status for a broken view. If previously a failure to prelock all functions used in a view would leave the connection in LTM_PRELOCKED mode, now we call close_thread_tables() from open_and_lock_tables() and leave prelocked mode, thus some check in mysql_admin_table() that works only in prelocked/locked tables mode is no longer activated. mysql-test/suite/rpl/r/rpl_row_implicit_commit_binlog.result: Fixed Bug#55452 "SET PASSWORD is replicated twice in RBR mode": extra binlog events are gone from the binary log. mysql-test/t/variables.test: Add a test case: set autocommit and statement transaction/prelocked mode. sql/event_data_objects.cc: Simplify code in Event_job_data::execute(). Move sp_head memory management to lex_end(). sql/event_db_repository.cc: Move the release of metadata locks outside close_thread_tables(). Make sure we call close_thread_tables() when open_and_lock_tables() fails and remove extra code from the events data dictionary. Use close_mysql_tables(), a new internal function to properly close mysql.* tables in the data dictionary. Contract Event_db_repository::drop_events_by_field, drop_schema_events into one function. When dropping all events in a schema, make sure we don't mistakenly release all locks acquired by DROP DATABASE. These include locks on the database name and the global intention exclusive metadata lock. sql/event_db_repository.h: Function open_event_table() does not require an instance of Event_db_repository. sql/events.cc: Use close_mysql_tables() instead of close_thread_tables() to bootstrap events, since the latter no longer releases metadata locks. sql/ha_ndbcluster.cc: - mysql_rm_table_part2 no longer releases acquired metadata locks. Do it in the caller. sql/ha_ndbcluster_binlog.cc: Deploy the new protocol for closing thread tables in run_query() and ndb_binlog_index code. sql/handler.cc: Assert that we never call ha_commit_trans/ ha_rollback_trans in sub-statement, which is now the case. sql/handler.h: Add an accessor to check whether THD_TRANS object is empty (has no transaction started). sql/log.cc: Update a comment. sql/log_event.cc: Since now we commit/rollback statement transaction in mysql_execute_command(), we need a mechanism to communicate from Query_log_event::do_apply_event() to mysql_execute_command() that the statement transaction should be rolled back, not committed. Ideally it would be a virtual method of THD. I hesitate to make THD a virtual base class in this already large patch. Use a thd->variables.option_bits for now. Remove a call to close_thread_tables() from the slave IO thread. It doesn't open any tables, and the protocol for closing thread tables is more complicated now. Make sure we properly close thread tables, however, in Load_data_log_event, which doesn't follow the standard server execution procedure with mysql_execute_command(). @todo: this piece should use Server_runnable framework instead. Remove an unnecessary call to mysql_unlock_tables(). sql/rpl_rli.cc: Update Relay_log_info::slave_close_thread_tables() to follow the new close protocol. sql/set_var.cc: Remove an unused header. sql/slave.cc: Remove an unnecessary call to close_thread_tables(). sql/sp.cc: Remove unnecessary calls to close_thread_tables() from SP DDL implementation. The tables will be closed by the caller, in mysql_execute_command(). When dropping all routines in a database, make sure to not mistakenly drop all metadata locks acquired so far, they include the scoped lock on the schema. sql/sp_head.cc: Correct the protocol that closes thread tables in an SP instruction. Clear lex->sphead before cleaning up lex with lex_end to make sure that we don't delete the sphead twice. It's considered to be "cleaner" and more in line with future changes than calling delete lex->sphead in other places that cleanup the lex. sql/sp_head.h: When destroying m_lex_keeper of an instruction, don't delete the sphead that all lex objects share. @todo: don't store a reference to routine's sp_head instance in instruction's lex. sql/sql_acl.cc: Don't call close_thread_tables() where the caller will do that for us. Fix Bug#55452 "SET PASSWORD is replicated twice in RBR mode" by disabling RBR replication in change_password() function. Use close_mysql_tables() in bootstrap and ACL reload code to make sure we release all metadata locks. sql/sql_base.cc: This is the main part of the patch: - remove manipulation with thd->transaction and thd->mdl_context from close_thread_tables(). Now this function is only responsible for closing tables, nothing else. This is necessary to be able to easily use close_thread_tables() in procedures, that involve multiple open/close tables, which all need to be protected continuously by metadata locks. Add asserts ensuring that TABLE object is only used when is protected by a metadata lock. Simplify the back off condition of Open_table_context, we no longer need to look at the autocommit mode. Make open_and_lock_tables() and open_normal_and_derived_tables() close thread tables and release metadata locks acquired so-far upon failure. This simplifies their usage. Implement close_mysql_tables(). sql/sql_base.h: Add declaration for close_mysql_tables(). sql/sql_class.cc: Remove a piece of dead code that has also become redundant after the fix for Bug 37521. The code became dead when my_eof() was made a non-protocol method, but a method that merely modifies the diagnostics area. The code became redundant with the fix for Bug#37521, when we started to cal close_thread_tables() before Protocol::end_statement(). sql/sql_do.cc: Do nothing in DO if inside a substatement (the assert moved out of trans_rollback_stmt). sql/sql_handler.cc: Add comments. sql/sql_insert.cc: Remove dead code. Release metadata locks explicitly at the end of the delayed insert thread. sql/sql_lex.cc: Add destruction of lex->sphead to lex_end(), lex "reset" method called at the end of each statement. sql/sql_parse.cc: Move close_thread_tables() and other related cleanups to mysql_execute_command() from dispatch_command(). This has become possible after the fix for Bug#37521. Mark federated SERVER statements as DDL. Next step: make sure that we don't store eof packet in the query cache, and move the query cache code outside mysql_parse. Brush up the code of COM_FIELD_LIST. Remove unnecessary calls to close_thread_tables(). When killing a query, don't report "OK" if it was a suicide. sql/sql_parse.h: Remove declaration of a function that is now static. sql/sql_partition.cc: Remove an unnecessary call to close_thread_tables(). sql/sql_plugin.cc: open_and_lock_tables() will clean up after itself after a failure. Move close_thread_tables() above end: label, and replace with close_mysql_tables(), which will also release the metadata lock on mysql.plugin. sql/sql_prepare.cc: Now that we no longer release locks in close_thread_tables() statement prepare code has become more straightforward. Remove the now redundant check for thd->killed() (used only by the backup project) from Execute_server_runnable. Reorder code to take into account that now mysql_execute_command() performs lex->unit.cleanup() and close_thread_tables(). sql/sql_priv.h: Add a new option to server options to interact between the slave SQL thread and execution framework (hack). @todo: use a virtual method of class THD instead. sql/sql_servers.cc: Due to Bug 25705 replication of DROP/CREATE/ALTER SERVER is broken. Make sure at least we do not attempt to replicate these statements using RBR, as this violates the assert in close_mysql_tables(). sql/sql_table.cc: Do not release metadata locks in mysql_rm_table_part2, this is done by the caller. Do not call close_thread_tables() in mysql_create_table(), this is done by the caller. Fix a bug in DROP TABLE under LOCK TABLES when, upon error in wait_while_table_is_used() we would mistakenly release the metadata lock on a non-dropped table. Explicitly release metadata locks when doing an implicit commit. sql/sql_trigger.cc: Now that we delete lex->sphead in lex_end(), zero the trigger's sphead in lex after loading the trigger, to avoid double deletion. sql/sql_udf.cc: Use close_mysql_tables() instead of close_thread_tables(). sql/sys_vars.cc: Remove code added in scope of WL#4284 which would break when we perform set @@session.autocommit along with setting other variables and using tables or functions. A test case added to variables.test. sql/transaction.cc: Add asserts. sql/tztime.cc: Use close_mysql_tables() rather than close_thread_tables().
1333 lines
38 KiB
C++
1333 lines
38 KiB
C++
/* Copyright (C) 2000-2003 MySQL AB, 2008-2009 Sun Microsystems, Inc
|
|
|
|
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., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA */
|
|
|
|
|
|
/*
|
|
The servers are saved in the system table "servers"
|
|
|
|
Currently, when the user performs an ALTER SERVER or a DROP SERVER
|
|
operation, it will cause all open tables which refer to the named
|
|
server connection to be flushed. This may cause some undesirable
|
|
behaviour with regard to currently running transactions. It is
|
|
expected that the DBA knows what s/he is doing when s/he performs
|
|
the ALTER SERVER or DROP SERVER operation.
|
|
|
|
TODO:
|
|
It is desirable for us to implement a callback mechanism instead where
|
|
callbacks can be registered for specific server protocols. The callback
|
|
will be fired when such a server name has been created/altered/dropped
|
|
or when statistics are to be gathered such as how many actual connections.
|
|
Storage engines etc will be able to make use of the callback so that
|
|
currently running transactions etc will not be disrupted.
|
|
*/
|
|
|
|
#include "sql_priv.h"
|
|
#include "sql_servers.h"
|
|
#include "unireg.h"
|
|
#include "sql_base.h" // close_mysql_tables
|
|
#include "records.h" // init_read_record, end_read_record
|
|
#include "hash_filo.h"
|
|
#include <m_ctype.h>
|
|
#include <stdarg.h>
|
|
#include "sp_head.h"
|
|
#include "sp.h"
|
|
#include "transaction.h"
|
|
#include "lock.h" // MYSQL_LOCK_IGNORE_TIMEOUT
|
|
|
|
/*
|
|
We only use 1 mutex to guard the data structures - THR_LOCK_servers.
|
|
Read locked when only reading data and write-locked for all other access.
|
|
*/
|
|
|
|
static HASH servers_cache;
|
|
static MEM_ROOT mem;
|
|
static mysql_rwlock_t THR_LOCK_servers;
|
|
|
|
static bool get_server_from_table_to_cache(TABLE *table);
|
|
|
|
/* insert functions */
|
|
static int insert_server(THD *thd, FOREIGN_SERVER *server_options);
|
|
static int insert_server_record(TABLE *table, FOREIGN_SERVER *server);
|
|
static int insert_server_record_into_cache(FOREIGN_SERVER *server);
|
|
static FOREIGN_SERVER *
|
|
prepare_server_struct_for_insert(LEX_SERVER_OPTIONS *server_options);
|
|
/* drop functions */
|
|
static int delete_server_record(TABLE *table,
|
|
char *server_name,
|
|
size_t server_name_length);
|
|
static int delete_server_record_in_cache(LEX_SERVER_OPTIONS *server_options);
|
|
|
|
/* update functions */
|
|
static void prepare_server_struct_for_update(LEX_SERVER_OPTIONS *server_options,
|
|
FOREIGN_SERVER *existing,
|
|
FOREIGN_SERVER *altered);
|
|
static int update_server(THD *thd, FOREIGN_SERVER *existing,
|
|
FOREIGN_SERVER *altered);
|
|
static int update_server_record(TABLE *table, FOREIGN_SERVER *server);
|
|
static int update_server_record_in_cache(FOREIGN_SERVER *existing,
|
|
FOREIGN_SERVER *altered);
|
|
/* utility functions */
|
|
static void merge_server_struct(FOREIGN_SERVER *from, FOREIGN_SERVER *to);
|
|
|
|
|
|
|
|
static uchar *servers_cache_get_key(FOREIGN_SERVER *server, size_t *length,
|
|
my_bool not_used __attribute__((unused)))
|
|
{
|
|
DBUG_ENTER("servers_cache_get_key");
|
|
DBUG_PRINT("info", ("server_name_length %d server_name %s",
|
|
server->server_name_length,
|
|
server->server_name));
|
|
|
|
*length= (uint) server->server_name_length;
|
|
DBUG_RETURN((uchar*) server->server_name);
|
|
}
|
|
|
|
#ifdef HAVE_PSI_INTERFACE
|
|
static PSI_rwlock_key key_rwlock_THR_LOCK_servers;
|
|
|
|
static PSI_rwlock_info all_servers_cache_rwlocks[]=
|
|
{
|
|
{ &key_rwlock_THR_LOCK_servers, "THR_LOCK_servers", PSI_FLAG_GLOBAL}
|
|
};
|
|
|
|
static void init_servers_cache_psi_keys(void)
|
|
{
|
|
const char* category= "sql";
|
|
int count;
|
|
|
|
if (PSI_server == NULL)
|
|
return;
|
|
|
|
count= array_elements(all_servers_cache_rwlocks);
|
|
PSI_server->register_rwlock(category, all_servers_cache_rwlocks, count);
|
|
}
|
|
#endif /* HAVE_PSI_INTERFACE */
|
|
|
|
/*
|
|
Initialize structures responsible for servers used in federated
|
|
server scheme information for them from the server
|
|
table in the 'mysql' database.
|
|
|
|
SYNOPSIS
|
|
servers_init()
|
|
dont_read_server_table TRUE if we want to skip loading data from
|
|
server table and disable privilege checking.
|
|
|
|
NOTES
|
|
This function is mostly responsible for preparatory steps, main work
|
|
on initialization and grants loading is done in servers_reload().
|
|
|
|
RETURN VALUES
|
|
0 ok
|
|
1 Could not initialize servers
|
|
*/
|
|
|
|
bool servers_init(bool dont_read_servers_table)
|
|
{
|
|
THD *thd;
|
|
bool return_val= FALSE;
|
|
DBUG_ENTER("servers_init");
|
|
|
|
#ifdef HAVE_PSI_INTERFACE
|
|
init_servers_cache_psi_keys();
|
|
#endif
|
|
|
|
/* init the mutex */
|
|
if (mysql_rwlock_init(key_rwlock_THR_LOCK_servers, &THR_LOCK_servers))
|
|
DBUG_RETURN(TRUE);
|
|
|
|
/* initialise our servers cache */
|
|
if (my_hash_init(&servers_cache, system_charset_info, 32, 0, 0,
|
|
(my_hash_get_key) servers_cache_get_key, 0, 0))
|
|
{
|
|
return_val= TRUE; /* we failed, out of memory? */
|
|
goto end;
|
|
}
|
|
|
|
/* Initialize the mem root for data */
|
|
init_sql_alloc(&mem, ACL_ALLOC_BLOCK_SIZE, 0);
|
|
|
|
if (dont_read_servers_table)
|
|
goto end;
|
|
|
|
/*
|
|
To be able to run this from boot, we allocate a temporary THD
|
|
*/
|
|
if (!(thd=new THD))
|
|
DBUG_RETURN(TRUE);
|
|
thd->thread_stack= (char*) &thd;
|
|
thd->store_globals();
|
|
/*
|
|
It is safe to call servers_reload() since servers_* arrays and hashes which
|
|
will be freed there are global static objects and thus are initialized
|
|
by zeros at startup.
|
|
*/
|
|
return_val= servers_reload(thd);
|
|
delete thd;
|
|
/* Remember that we don't have a THD */
|
|
my_pthread_setspecific_ptr(THR_THD, 0);
|
|
|
|
end:
|
|
DBUG_RETURN(return_val);
|
|
}
|
|
|
|
/*
|
|
Initialize server structures
|
|
|
|
SYNOPSIS
|
|
servers_load()
|
|
thd Current thread
|
|
tables List containing open "mysql.servers"
|
|
|
|
RETURN VALUES
|
|
FALSE Success
|
|
TRUE Error
|
|
|
|
TODO
|
|
Revert back to old list if we failed to load new one.
|
|
*/
|
|
|
|
static bool servers_load(THD *thd, TABLE_LIST *tables)
|
|
{
|
|
TABLE *table;
|
|
READ_RECORD read_record_info;
|
|
bool return_val= TRUE;
|
|
DBUG_ENTER("servers_load");
|
|
|
|
my_hash_reset(&servers_cache);
|
|
free_root(&mem, MYF(0));
|
|
init_sql_alloc(&mem, ACL_ALLOC_BLOCK_SIZE, 0);
|
|
|
|
init_read_record(&read_record_info,thd,table=tables[0].table,NULL,1,0,
|
|
FALSE);
|
|
while (!(read_record_info.read_record(&read_record_info)))
|
|
{
|
|
/* return_val is already TRUE, so no need to set */
|
|
if ((get_server_from_table_to_cache(table)))
|
|
goto end;
|
|
}
|
|
|
|
return_val= FALSE;
|
|
|
|
end:
|
|
end_read_record(&read_record_info);
|
|
DBUG_RETURN(return_val);
|
|
}
|
|
|
|
|
|
/*
|
|
Forget current servers cache and read new servers
|
|
from the conneciton table.
|
|
|
|
SYNOPSIS
|
|
servers_reload()
|
|
thd Current thread
|
|
|
|
NOTE
|
|
All tables of calling thread which were open and locked by LOCK TABLES
|
|
statement will be unlocked and closed.
|
|
This function is also used for initialization of structures responsible
|
|
for user/db-level privilege checking.
|
|
|
|
RETURN VALUE
|
|
FALSE Success
|
|
TRUE Failure
|
|
*/
|
|
|
|
bool servers_reload(THD *thd)
|
|
{
|
|
TABLE_LIST tables[1];
|
|
bool return_val= TRUE;
|
|
DBUG_ENTER("servers_reload");
|
|
|
|
DBUG_PRINT("info", ("locking servers_cache"));
|
|
mysql_rwlock_wrlock(&THR_LOCK_servers);
|
|
|
|
tables[0].init_one_table("mysql", 5, "servers", 7, "servers", TL_READ);
|
|
|
|
if (open_and_lock_tables(thd, tables, FALSE, MYSQL_LOCK_IGNORE_TIMEOUT))
|
|
{
|
|
/*
|
|
Execution might have been interrupted; only print the error message
|
|
if an error condition has been raised.
|
|
*/
|
|
if (thd->stmt_da->is_error())
|
|
sql_print_error("Can't open and lock privilege tables: %s",
|
|
thd->stmt_da->message());
|
|
return_val= FALSE;
|
|
goto end;
|
|
}
|
|
|
|
if ((return_val= servers_load(thd, tables)))
|
|
{ // Error. Revert to old list
|
|
/* blast, for now, we have no servers, discuss later way to preserve */
|
|
|
|
DBUG_PRINT("error",("Reverting to old privileges"));
|
|
servers_free();
|
|
}
|
|
|
|
end:
|
|
close_mysql_tables(thd);
|
|
DBUG_PRINT("info", ("unlocking servers_cache"));
|
|
mysql_rwlock_unlock(&THR_LOCK_servers);
|
|
DBUG_RETURN(return_val);
|
|
}
|
|
|
|
|
|
/*
|
|
Initialize structures responsible for servers used in federated
|
|
server scheme information for them from the server
|
|
table in the 'mysql' database.
|
|
|
|
SYNOPSIS
|
|
get_server_from_table_to_cache()
|
|
TABLE *table open table pointer
|
|
|
|
|
|
NOTES
|
|
This function takes a TABLE pointer (pointing to an opened
|
|
table). With this open table, a FOREIGN_SERVER struct pointer
|
|
is allocated into root memory, then each member of the FOREIGN_SERVER
|
|
struct is populated. A char pointer takes the return value of get_field
|
|
for each column we're interested in obtaining, and if that pointer
|
|
isn't 0x0, the FOREIGN_SERVER member is set to that value, otherwise,
|
|
is set to the value of an empty string, since get_field would set it to
|
|
0x0 if the column's value is empty, even if the default value for that
|
|
column is NOT NULL.
|
|
|
|
RETURN VALUES
|
|
0 ok
|
|
1 could not insert server struct into global servers cache
|
|
*/
|
|
|
|
static bool
|
|
get_server_from_table_to_cache(TABLE *table)
|
|
{
|
|
/* alloc a server struct */
|
|
char *ptr;
|
|
char * const blank= (char*)"";
|
|
FOREIGN_SERVER *server= (FOREIGN_SERVER *)alloc_root(&mem,
|
|
sizeof(FOREIGN_SERVER));
|
|
DBUG_ENTER("get_server_from_table_to_cache");
|
|
table->use_all_columns();
|
|
|
|
/* get each field into the server struct ptr */
|
|
server->server_name= get_field(&mem, table->field[0]);
|
|
server->server_name_length= (uint) strlen(server->server_name);
|
|
ptr= get_field(&mem, table->field[1]);
|
|
server->host= ptr ? ptr : blank;
|
|
ptr= get_field(&mem, table->field[2]);
|
|
server->db= ptr ? ptr : blank;
|
|
ptr= get_field(&mem, table->field[3]);
|
|
server->username= ptr ? ptr : blank;
|
|
ptr= get_field(&mem, table->field[4]);
|
|
server->password= ptr ? ptr : blank;
|
|
ptr= get_field(&mem, table->field[5]);
|
|
server->sport= ptr ? ptr : blank;
|
|
|
|
server->port= server->sport ? atoi(server->sport) : 0;
|
|
|
|
ptr= get_field(&mem, table->field[6]);
|
|
server->socket= ptr && strlen(ptr) ? ptr : blank;
|
|
ptr= get_field(&mem, table->field[7]);
|
|
server->scheme= ptr ? ptr : blank;
|
|
ptr= get_field(&mem, table->field[8]);
|
|
server->owner= ptr ? ptr : blank;
|
|
DBUG_PRINT("info", ("server->server_name %s", server->server_name));
|
|
DBUG_PRINT("info", ("server->host %s", server->host));
|
|
DBUG_PRINT("info", ("server->db %s", server->db));
|
|
DBUG_PRINT("info", ("server->username %s", server->username));
|
|
DBUG_PRINT("info", ("server->password %s", server->password));
|
|
DBUG_PRINT("info", ("server->socket %s", server->socket));
|
|
if (my_hash_insert(&servers_cache, (uchar*) server))
|
|
{
|
|
DBUG_PRINT("info", ("had a problem inserting server %s at %lx",
|
|
server->server_name, (long unsigned int) server));
|
|
// error handling needed here
|
|
DBUG_RETURN(TRUE);
|
|
}
|
|
DBUG_RETURN(FALSE);
|
|
}
|
|
|
|
|
|
/*
|
|
SYNOPSIS
|
|
insert_server()
|
|
THD *thd - thread pointer
|
|
FOREIGN_SERVER *server - pointer to prepared FOREIGN_SERVER struct
|
|
|
|
NOTES
|
|
This function takes a server object that is has all members properly
|
|
prepared, ready to be inserted both into the mysql.servers table and
|
|
the servers cache.
|
|
|
|
THR_LOCK_servers must be write locked.
|
|
|
|
RETURN VALUES
|
|
0 - no error
|
|
other - error code
|
|
*/
|
|
|
|
static int
|
|
insert_server(THD *thd, FOREIGN_SERVER *server)
|
|
{
|
|
int error= -1;
|
|
TABLE_LIST tables;
|
|
TABLE *table;
|
|
|
|
DBUG_ENTER("insert_server");
|
|
|
|
tables.init_one_table("mysql", 5, "servers", 7, "servers", TL_WRITE);
|
|
|
|
/* need to open before acquiring THR_LOCK_plugin or it will deadlock */
|
|
if (! (table= open_ltable(thd, &tables, TL_WRITE, MYSQL_LOCK_IGNORE_TIMEOUT)))
|
|
goto end;
|
|
|
|
/* insert the server into the table */
|
|
if ((error= insert_server_record(table, server)))
|
|
goto end;
|
|
|
|
/* insert the server into the cache */
|
|
if ((error= insert_server_record_into_cache(server)))
|
|
goto end;
|
|
|
|
end:
|
|
DBUG_RETURN(error);
|
|
}
|
|
|
|
|
|
/*
|
|
SYNOPSIS
|
|
int insert_server_record_into_cache()
|
|
FOREIGN_SERVER *server
|
|
|
|
NOTES
|
|
This function takes a FOREIGN_SERVER pointer to an allocated (root mem)
|
|
and inserts it into the global servers cache
|
|
|
|
THR_LOCK_servers must be write locked.
|
|
|
|
RETURN VALUE
|
|
0 - no error
|
|
>0 - error code
|
|
|
|
*/
|
|
|
|
static int
|
|
insert_server_record_into_cache(FOREIGN_SERVER *server)
|
|
{
|
|
int error=0;
|
|
DBUG_ENTER("insert_server_record_into_cache");
|
|
/*
|
|
We succeded in insertion of the server to the table, now insert
|
|
the server to the cache
|
|
*/
|
|
DBUG_PRINT("info", ("inserting server %s at %lx, length %d",
|
|
server->server_name, (long unsigned int) server,
|
|
server->server_name_length));
|
|
if (my_hash_insert(&servers_cache, (uchar*) server))
|
|
{
|
|
DBUG_PRINT("info", ("had a problem inserting server %s at %lx",
|
|
server->server_name, (long unsigned int) server));
|
|
// error handling needed here
|
|
error= 1;
|
|
}
|
|
DBUG_RETURN(error);
|
|
}
|
|
|
|
|
|
/*
|
|
SYNOPSIS
|
|
store_server_fields()
|
|
TABLE *table
|
|
FOREIGN_SERVER *server
|
|
|
|
NOTES
|
|
This function takes an opened table object, and a pointer to an
|
|
allocated FOREIGN_SERVER struct, and then stores each member of
|
|
the FOREIGN_SERVER to the appropriate fields in the table, in
|
|
advance of insertion into the mysql.servers table
|
|
|
|
RETURN VALUE
|
|
VOID
|
|
|
|
*/
|
|
|
|
static void
|
|
store_server_fields(TABLE *table, FOREIGN_SERVER *server)
|
|
{
|
|
|
|
table->use_all_columns();
|
|
/*
|
|
"server" has already been prepped by prepare_server_struct_for_<>
|
|
so, all we need to do is check if the value is set (> -1 for port)
|
|
|
|
If this happens to be an update, only the server members that
|
|
have changed will be set. If an insert, then all will be set,
|
|
even if with empty strings
|
|
*/
|
|
if (server->host)
|
|
table->field[1]->store(server->host,
|
|
(uint) strlen(server->host), system_charset_info);
|
|
if (server->db)
|
|
table->field[2]->store(server->db,
|
|
(uint) strlen(server->db), system_charset_info);
|
|
if (server->username)
|
|
table->field[3]->store(server->username,
|
|
(uint) strlen(server->username), system_charset_info);
|
|
if (server->password)
|
|
table->field[4]->store(server->password,
|
|
(uint) strlen(server->password), system_charset_info);
|
|
if (server->port > -1)
|
|
table->field[5]->store(server->port);
|
|
|
|
if (server->socket)
|
|
table->field[6]->store(server->socket,
|
|
(uint) strlen(server->socket), system_charset_info);
|
|
if (server->scheme)
|
|
table->field[7]->store(server->scheme,
|
|
(uint) strlen(server->scheme), system_charset_info);
|
|
if (server->owner)
|
|
table->field[8]->store(server->owner,
|
|
(uint) strlen(server->owner), system_charset_info);
|
|
}
|
|
|
|
/*
|
|
SYNOPSIS
|
|
insert_server_record()
|
|
TABLE *table
|
|
FOREIGN_SERVER *server
|
|
|
|
NOTES
|
|
This function takes the arguments of an open table object and a pointer
|
|
to an allocated FOREIGN_SERVER struct. It stores the server_name into
|
|
the first field of the table (the primary key, server_name column). With
|
|
this, index_read_idx is called, if the record is found, an error is set
|
|
to ER_FOREIGN_SERVER_EXISTS (the server with that server name exists in the
|
|
table), if not, then store_server_fields stores all fields of the
|
|
FOREIGN_SERVER to the table, then ha_write_row is inserted. If an error
|
|
is encountered in either index_read_idx or ha_write_row, then that error
|
|
is returned
|
|
|
|
RETURN VALUE
|
|
0 - no errors
|
|
>0 - error code
|
|
|
|
*/
|
|
|
|
static
|
|
int insert_server_record(TABLE *table, FOREIGN_SERVER *server)
|
|
{
|
|
int error;
|
|
DBUG_ENTER("insert_server_record");
|
|
tmp_disable_binlog(table->in_use);
|
|
table->use_all_columns();
|
|
|
|
empty_record(table);
|
|
|
|
/* set the field that's the PK to the value we're looking for */
|
|
table->field[0]->store(server->server_name,
|
|
server->server_name_length,
|
|
system_charset_info);
|
|
|
|
/* read index until record is that specified in server_name */
|
|
if ((error= table->file->index_read_idx_map(table->record[0], 0,
|
|
(uchar *)table->field[0]->ptr,
|
|
HA_WHOLE_KEY,
|
|
HA_READ_KEY_EXACT)))
|
|
{
|
|
/* if not found, err */
|
|
if (error != HA_ERR_KEY_NOT_FOUND && error != HA_ERR_END_OF_FILE)
|
|
{
|
|
table->file->print_error(error, MYF(0));
|
|
error= 1;
|
|
}
|
|
/* store each field to be inserted */
|
|
store_server_fields(table, server);
|
|
|
|
DBUG_PRINT("info",("record for server '%s' not found!",
|
|
server->server_name));
|
|
/* write/insert the new server */
|
|
if ((error=table->file->ha_write_row(table->record[0])))
|
|
{
|
|
table->file->print_error(error, MYF(0));
|
|
}
|
|
else
|
|
error= 0;
|
|
}
|
|
else
|
|
error= ER_FOREIGN_SERVER_EXISTS;
|
|
|
|
reenable_binlog(table->in_use);
|
|
DBUG_RETURN(error);
|
|
}
|
|
|
|
/*
|
|
SYNOPSIS
|
|
drop_server()
|
|
THD *thd
|
|
LEX_SERVER_OPTIONS *server_options
|
|
|
|
NOTES
|
|
This function takes as its arguments a THD object pointer and a pointer
|
|
to a LEX_SERVER_OPTIONS struct from the parser. The member 'server_name'
|
|
of this LEX_SERVER_OPTIONS struct contains the value of the server to be
|
|
deleted. The mysql.servers table is opened via open_ltable, a table object
|
|
returned, the servers cache mutex locked, then delete_server_record is
|
|
called with this table object and LEX_SERVER_OPTIONS server_name and
|
|
server_name_length passed, containing the name of the server to be
|
|
dropped/deleted, then delete_server_record_in_cache is called to delete
|
|
the server from the servers cache.
|
|
|
|
RETURN VALUE
|
|
0 - no error
|
|
> 0 - error code
|
|
*/
|
|
|
|
int drop_server(THD *thd, LEX_SERVER_OPTIONS *server_options)
|
|
{
|
|
int error;
|
|
TABLE_LIST tables;
|
|
TABLE *table;
|
|
LEX_STRING name= { server_options->server_name,
|
|
server_options->server_name_length };
|
|
|
|
DBUG_ENTER("drop_server");
|
|
DBUG_PRINT("info", ("server name server->server_name %s",
|
|
server_options->server_name));
|
|
|
|
tables.init_one_table("mysql", 5, "servers", 7, "servers", TL_WRITE);
|
|
|
|
mysql_rwlock_wrlock(&THR_LOCK_servers);
|
|
|
|
/* hit the memory hit first */
|
|
if ((error= delete_server_record_in_cache(server_options)))
|
|
goto end;
|
|
|
|
if (! (table= open_ltable(thd, &tables, TL_WRITE, MYSQL_LOCK_IGNORE_TIMEOUT)))
|
|
{
|
|
error= my_errno;
|
|
goto end;
|
|
}
|
|
|
|
error= delete_server_record(table, name.str, name.length);
|
|
|
|
/* close the servers table before we call closed_cached_connection_tables */
|
|
close_mysql_tables(thd);
|
|
|
|
if (close_cached_connection_tables(thd, TRUE, &name))
|
|
{
|
|
push_warning_printf(thd, MYSQL_ERROR::WARN_LEVEL_WARN,
|
|
ER_UNKNOWN_ERROR, "Server connection in use");
|
|
}
|
|
|
|
end:
|
|
mysql_rwlock_unlock(&THR_LOCK_servers);
|
|
DBUG_RETURN(error);
|
|
}
|
|
|
|
|
|
/*
|
|
|
|
SYNOPSIS
|
|
delete_server_record_in_cache()
|
|
LEX_SERVER_OPTIONS *server_options
|
|
|
|
NOTES
|
|
This function's argument is a LEX_SERVER_OPTIONS struct pointer. This
|
|
function uses the "server_name" and "server_name_length" members of the
|
|
lex->server_options to search for the server in the servers_cache. Upon
|
|
returned the server (pointer to a FOREIGN_SERVER struct), it then deletes
|
|
that server from the servers_cache hash.
|
|
|
|
RETURN VALUE
|
|
0 - no error
|
|
|
|
*/
|
|
|
|
static int
|
|
delete_server_record_in_cache(LEX_SERVER_OPTIONS *server_options)
|
|
{
|
|
int error= ER_FOREIGN_SERVER_DOESNT_EXIST;
|
|
FOREIGN_SERVER *server;
|
|
DBUG_ENTER("delete_server_record_in_cache");
|
|
|
|
DBUG_PRINT("info",("trying to obtain server name %s length %d",
|
|
server_options->server_name,
|
|
server_options->server_name_length));
|
|
|
|
|
|
if (!(server= (FOREIGN_SERVER *)
|
|
my_hash_search(&servers_cache,
|
|
(uchar*) server_options->server_name,
|
|
server_options->server_name_length)))
|
|
{
|
|
DBUG_PRINT("info", ("server_name %s length %d not found!",
|
|
server_options->server_name,
|
|
server_options->server_name_length));
|
|
goto end;
|
|
}
|
|
/*
|
|
We succeded in deletion of the server to the table, now delete
|
|
the server from the cache
|
|
*/
|
|
DBUG_PRINT("info",("deleting server %s length %d",
|
|
server->server_name,
|
|
server->server_name_length));
|
|
|
|
my_hash_delete(&servers_cache, (uchar*) server);
|
|
|
|
error= 0;
|
|
|
|
end:
|
|
DBUG_RETURN(error);
|
|
}
|
|
|
|
|
|
/*
|
|
|
|
SYNOPSIS
|
|
update_server()
|
|
THD *thd
|
|
FOREIGN_SERVER *existing
|
|
FOREIGN_SERVER *altered
|
|
|
|
NOTES
|
|
This function takes as arguments a THD object pointer, and two pointers,
|
|
one pointing to the existing FOREIGN_SERVER struct "existing" (which is
|
|
the current record as it is) and another pointer pointing to the
|
|
FOREIGN_SERVER struct with the members containing the modified/altered
|
|
values that need to be updated in both the mysql.servers table and the
|
|
servers_cache. It opens a table, passes the table and the altered
|
|
FOREIGN_SERVER pointer, which will be used to update the mysql.servers
|
|
table for the particular server via the call to update_server_record,
|
|
and in the servers_cache via update_server_record_in_cache.
|
|
|
|
THR_LOCK_servers must be write locked.
|
|
|
|
RETURN VALUE
|
|
0 - no error
|
|
>0 - error code
|
|
|
|
*/
|
|
|
|
int update_server(THD *thd, FOREIGN_SERVER *existing, FOREIGN_SERVER *altered)
|
|
{
|
|
int error;
|
|
TABLE *table;
|
|
TABLE_LIST tables;
|
|
DBUG_ENTER("update_server");
|
|
|
|
tables.init_one_table("mysql", 5, "servers", 7, "servers",
|
|
TL_WRITE);
|
|
|
|
if (!(table= open_ltable(thd, &tables, TL_WRITE, MYSQL_LOCK_IGNORE_TIMEOUT)))
|
|
{
|
|
error= my_errno;
|
|
goto end;
|
|
}
|
|
|
|
if ((error= update_server_record(table, altered)))
|
|
goto end;
|
|
|
|
error= update_server_record_in_cache(existing, altered);
|
|
|
|
/*
|
|
Perform a reload so we don't have a 'hole' in our mem_root
|
|
*/
|
|
servers_load(thd, &tables);
|
|
|
|
end:
|
|
DBUG_RETURN(error);
|
|
}
|
|
|
|
|
|
/*
|
|
|
|
SYNOPSIS
|
|
update_server_record_in_cache()
|
|
FOREIGN_SERVER *existing
|
|
FOREIGN_SERVER *altered
|
|
|
|
NOTES
|
|
This function takes as an argument the FOREIGN_SERVER structi pointer
|
|
for the existing server and the FOREIGN_SERVER struct populated with only
|
|
the members which have been updated. It then "merges" the "altered" struct
|
|
members to the existing server, the existing server then represents an
|
|
updated server. Then, the existing record is deleted from the servers_cache
|
|
HASH, then the updated record inserted, in essence replacing the old
|
|
record.
|
|
|
|
THR_LOCK_servers must be write locked.
|
|
|
|
RETURN VALUE
|
|
0 - no error
|
|
1 - error
|
|
|
|
*/
|
|
|
|
int update_server_record_in_cache(FOREIGN_SERVER *existing,
|
|
FOREIGN_SERVER *altered)
|
|
{
|
|
int error= 0;
|
|
DBUG_ENTER("update_server_record_in_cache");
|
|
|
|
/*
|
|
update the members that haven't been change in the altered server struct
|
|
with the values of the existing server struct
|
|
*/
|
|
merge_server_struct(existing, altered);
|
|
|
|
/*
|
|
delete the existing server struct from the server cache
|
|
*/
|
|
my_hash_delete(&servers_cache, (uchar*)existing);
|
|
|
|
/*
|
|
Insert the altered server struct into the server cache
|
|
*/
|
|
if (my_hash_insert(&servers_cache, (uchar*)altered))
|
|
{
|
|
DBUG_PRINT("info", ("had a problem inserting server %s at %lx",
|
|
altered->server_name, (long unsigned int) altered));
|
|
error= ER_OUT_OF_RESOURCES;
|
|
}
|
|
|
|
DBUG_RETURN(error);
|
|
}
|
|
|
|
|
|
/*
|
|
|
|
SYNOPSIS
|
|
merge_server_struct()
|
|
FOREIGN_SERVER *from
|
|
FOREIGN_SERVER *to
|
|
|
|
NOTES
|
|
This function takes as its arguments two pointers each to an allocated
|
|
FOREIGN_SERVER struct. The first FOREIGN_SERVER struct represents the struct
|
|
that we will obtain values from (hence the name "from"), the second
|
|
FOREIGN_SERVER struct represents which FOREIGN_SERVER struct we will be
|
|
"copying" any members that have a value to (hence the name "to")
|
|
|
|
RETURN VALUE
|
|
VOID
|
|
|
|
*/
|
|
|
|
void merge_server_struct(FOREIGN_SERVER *from, FOREIGN_SERVER *to)
|
|
{
|
|
DBUG_ENTER("merge_server_struct");
|
|
if (!to->host)
|
|
to->host= strdup_root(&mem, from->host);
|
|
if (!to->db)
|
|
to->db= strdup_root(&mem, from->db);
|
|
if (!to->username)
|
|
to->username= strdup_root(&mem, from->username);
|
|
if (!to->password)
|
|
to->password= strdup_root(&mem, from->password);
|
|
if (to->port == -1)
|
|
to->port= from->port;
|
|
if (!to->socket && from->socket)
|
|
to->socket= strdup_root(&mem, from->socket);
|
|
if (!to->scheme && from->scheme)
|
|
to->scheme= strdup_root(&mem, from->scheme);
|
|
if (!to->owner)
|
|
to->owner= strdup_root(&mem, from->owner);
|
|
|
|
DBUG_VOID_RETURN;
|
|
}
|
|
|
|
|
|
/*
|
|
|
|
SYNOPSIS
|
|
update_server_record()
|
|
TABLE *table
|
|
FOREIGN_SERVER *server
|
|
|
|
NOTES
|
|
This function takes as its arguments an open TABLE pointer, and a pointer
|
|
to an allocated FOREIGN_SERVER structure representing an updated record
|
|
which needs to be inserted. The primary key, server_name is stored to field
|
|
0, then index_read_idx is called to read the index to that record, the
|
|
record then being ready to be updated, if found. If not found an error is
|
|
set and error message printed. If the record is found, store_record is
|
|
called, then store_server_fields stores each field from the the members of
|
|
the updated FOREIGN_SERVER struct.
|
|
|
|
RETURN VALUE
|
|
0 - no error
|
|
|
|
*/
|
|
|
|
|
|
static int
|
|
update_server_record(TABLE *table, FOREIGN_SERVER *server)
|
|
{
|
|
int error=0;
|
|
DBUG_ENTER("update_server_record");
|
|
tmp_disable_binlog(table->in_use);
|
|
table->use_all_columns();
|
|
/* set the field that's the PK to the value we're looking for */
|
|
table->field[0]->store(server->server_name,
|
|
server->server_name_length,
|
|
system_charset_info);
|
|
|
|
if ((error= table->file->index_read_idx_map(table->record[0], 0,
|
|
(uchar *)table->field[0]->ptr,
|
|
~(longlong)0,
|
|
HA_READ_KEY_EXACT)))
|
|
{
|
|
if (error != HA_ERR_KEY_NOT_FOUND && error != HA_ERR_END_OF_FILE)
|
|
table->file->print_error(error, MYF(0));
|
|
DBUG_PRINT("info",("server not found!"));
|
|
error= ER_FOREIGN_SERVER_DOESNT_EXIST;
|
|
}
|
|
else
|
|
{
|
|
/* ok, so we can update since the record exists in the table */
|
|
store_record(table,record[1]);
|
|
store_server_fields(table, server);
|
|
if ((error=table->file->ha_update_row(table->record[1],
|
|
table->record[0])) &&
|
|
error != HA_ERR_RECORD_IS_THE_SAME)
|
|
{
|
|
DBUG_PRINT("info",("problems with ha_update_row %d", error));
|
|
goto end;
|
|
}
|
|
else
|
|
error= 0;
|
|
}
|
|
|
|
end:
|
|
reenable_binlog(table->in_use);
|
|
DBUG_RETURN(error);
|
|
}
|
|
|
|
|
|
/*
|
|
|
|
SYNOPSIS
|
|
delete_server_record()
|
|
TABLE *table
|
|
char *server_name
|
|
int server_name_length
|
|
|
|
NOTES
|
|
|
|
RETURN VALUE
|
|
0 - no error
|
|
|
|
*/
|
|
|
|
static int
|
|
delete_server_record(TABLE *table,
|
|
char *server_name, size_t server_name_length)
|
|
{
|
|
int error;
|
|
DBUG_ENTER("delete_server_record");
|
|
tmp_disable_binlog(table->in_use);
|
|
table->use_all_columns();
|
|
|
|
/* set the field that's the PK to the value we're looking for */
|
|
table->field[0]->store(server_name, server_name_length, system_charset_info);
|
|
|
|
if ((error= table->file->index_read_idx_map(table->record[0], 0,
|
|
(uchar *)table->field[0]->ptr,
|
|
HA_WHOLE_KEY,
|
|
HA_READ_KEY_EXACT)))
|
|
{
|
|
if (error != HA_ERR_KEY_NOT_FOUND && error != HA_ERR_END_OF_FILE)
|
|
table->file->print_error(error, MYF(0));
|
|
DBUG_PRINT("info",("server not found!"));
|
|
error= ER_FOREIGN_SERVER_DOESNT_EXIST;
|
|
}
|
|
else
|
|
{
|
|
if ((error= table->file->ha_delete_row(table->record[0])))
|
|
table->file->print_error(error, MYF(0));
|
|
}
|
|
|
|
reenable_binlog(table->in_use);
|
|
DBUG_RETURN(error);
|
|
}
|
|
|
|
/*
|
|
|
|
SYNOPSIS
|
|
create_server()
|
|
THD *thd
|
|
LEX_SERVER_OPTIONS *server_options
|
|
|
|
NOTES
|
|
|
|
RETURN VALUE
|
|
0 - no error
|
|
|
|
*/
|
|
|
|
int create_server(THD *thd, LEX_SERVER_OPTIONS *server_options)
|
|
{
|
|
int error= ER_FOREIGN_SERVER_EXISTS;
|
|
FOREIGN_SERVER *server;
|
|
|
|
DBUG_ENTER("create_server");
|
|
DBUG_PRINT("info", ("server_options->server_name %s",
|
|
server_options->server_name));
|
|
|
|
mysql_rwlock_wrlock(&THR_LOCK_servers);
|
|
|
|
/* hit the memory first */
|
|
if (my_hash_search(&servers_cache, (uchar*) server_options->server_name,
|
|
server_options->server_name_length))
|
|
goto end;
|
|
|
|
|
|
if (!(server= prepare_server_struct_for_insert(server_options)))
|
|
{
|
|
/* purecov: begin inspected */
|
|
error= ER_OUT_OF_RESOURCES;
|
|
goto end;
|
|
/* purecov: end */
|
|
}
|
|
|
|
error= insert_server(thd, server);
|
|
|
|
DBUG_PRINT("info", ("error returned %d", error));
|
|
|
|
end:
|
|
mysql_rwlock_unlock(&THR_LOCK_servers);
|
|
DBUG_RETURN(error);
|
|
}
|
|
|
|
|
|
/*
|
|
|
|
SYNOPSIS
|
|
alter_server()
|
|
THD *thd
|
|
LEX_SERVER_OPTIONS *server_options
|
|
|
|
NOTES
|
|
|
|
RETURN VALUE
|
|
0 - no error
|
|
|
|
*/
|
|
|
|
int alter_server(THD *thd, LEX_SERVER_OPTIONS *server_options)
|
|
{
|
|
int error= ER_FOREIGN_SERVER_DOESNT_EXIST;
|
|
FOREIGN_SERVER *altered, *existing;
|
|
LEX_STRING name= { server_options->server_name,
|
|
server_options->server_name_length };
|
|
DBUG_ENTER("alter_server");
|
|
DBUG_PRINT("info", ("server_options->server_name %s",
|
|
server_options->server_name));
|
|
|
|
mysql_rwlock_wrlock(&THR_LOCK_servers);
|
|
|
|
if (!(existing= (FOREIGN_SERVER *) my_hash_search(&servers_cache,
|
|
(uchar*) name.str,
|
|
name.length)))
|
|
goto end;
|
|
|
|
altered= (FOREIGN_SERVER *)alloc_root(&mem,
|
|
sizeof(FOREIGN_SERVER));
|
|
|
|
prepare_server_struct_for_update(server_options, existing, altered);
|
|
|
|
error= update_server(thd, existing, altered);
|
|
|
|
/* close the servers table before we call closed_cached_connection_tables */
|
|
close_mysql_tables(thd);
|
|
|
|
if (close_cached_connection_tables(thd, FALSE, &name))
|
|
{
|
|
push_warning_printf(thd, MYSQL_ERROR::WARN_LEVEL_WARN,
|
|
ER_UNKNOWN_ERROR, "Server connection in use");
|
|
}
|
|
|
|
end:
|
|
DBUG_PRINT("info", ("error returned %d", error));
|
|
mysql_rwlock_unlock(&THR_LOCK_servers);
|
|
DBUG_RETURN(error);
|
|
}
|
|
|
|
|
|
/*
|
|
|
|
SYNOPSIS
|
|
prepare_server_struct_for_insert()
|
|
LEX_SERVER_OPTIONS *server_options
|
|
|
|
NOTES
|
|
As FOREIGN_SERVER members are allocated on mem_root, we do not need to
|
|
free them in case of error.
|
|
|
|
RETURN VALUE
|
|
On success filled FOREIGN_SERVER, or NULL in case out of memory.
|
|
|
|
*/
|
|
|
|
static FOREIGN_SERVER *
|
|
prepare_server_struct_for_insert(LEX_SERVER_OPTIONS *server_options)
|
|
{
|
|
char *unset_ptr= (char*)"";
|
|
FOREIGN_SERVER *server;
|
|
DBUG_ENTER("prepare_server_struct");
|
|
|
|
if (!(server= (FOREIGN_SERVER *)alloc_root(&mem, sizeof(FOREIGN_SERVER))))
|
|
DBUG_RETURN(NULL); /* purecov: inspected */
|
|
|
|
/* these two MUST be set */
|
|
if (!(server->server_name= strdup_root(&mem, server_options->server_name)))
|
|
DBUG_RETURN(NULL); /* purecov: inspected */
|
|
server->server_name_length= server_options->server_name_length;
|
|
|
|
if (!(server->host= server_options->host ?
|
|
strdup_root(&mem, server_options->host) : unset_ptr))
|
|
DBUG_RETURN(NULL); /* purecov: inspected */
|
|
|
|
if (!(server->db= server_options->db ?
|
|
strdup_root(&mem, server_options->db) : unset_ptr))
|
|
DBUG_RETURN(NULL); /* purecov: inspected */
|
|
|
|
if (!(server->username= server_options->username ?
|
|
strdup_root(&mem, server_options->username) : unset_ptr))
|
|
DBUG_RETURN(NULL); /* purecov: inspected */
|
|
|
|
if (!(server->password= server_options->password ?
|
|
strdup_root(&mem, server_options->password) : unset_ptr))
|
|
DBUG_RETURN(NULL); /* purecov: inspected */
|
|
|
|
/* set to 0 if not specified */
|
|
server->port= server_options->port > -1 ?
|
|
server_options->port : 0;
|
|
|
|
if (!(server->socket= server_options->socket ?
|
|
strdup_root(&mem, server_options->socket) : unset_ptr))
|
|
DBUG_RETURN(NULL); /* purecov: inspected */
|
|
|
|
if (!(server->scheme= server_options->scheme ?
|
|
strdup_root(&mem, server_options->scheme) : unset_ptr))
|
|
DBUG_RETURN(NULL); /* purecov: inspected */
|
|
|
|
if (!(server->owner= server_options->owner ?
|
|
strdup_root(&mem, server_options->owner) : unset_ptr))
|
|
DBUG_RETURN(NULL); /* purecov: inspected */
|
|
|
|
DBUG_RETURN(server);
|
|
}
|
|
|
|
/*
|
|
|
|
SYNOPSIS
|
|
prepare_server_struct_for_update()
|
|
LEX_SERVER_OPTIONS *server_options
|
|
|
|
NOTES
|
|
|
|
RETURN VALUE
|
|
0 - no error
|
|
|
|
*/
|
|
|
|
static void
|
|
prepare_server_struct_for_update(LEX_SERVER_OPTIONS *server_options,
|
|
FOREIGN_SERVER *existing,
|
|
FOREIGN_SERVER *altered)
|
|
{
|
|
DBUG_ENTER("prepare_server_struct_for_update");
|
|
|
|
altered->server_name= strdup_root(&mem, server_options->server_name);
|
|
altered->server_name_length= server_options->server_name_length;
|
|
DBUG_PRINT("info", ("existing name %s altered name %s",
|
|
existing->server_name, altered->server_name));
|
|
|
|
/*
|
|
The logic here is this: is this value set AND is it different
|
|
than the existing value?
|
|
*/
|
|
altered->host=
|
|
(server_options->host && (strcmp(server_options->host, existing->host))) ?
|
|
strdup_root(&mem, server_options->host) : 0;
|
|
|
|
altered->db=
|
|
(server_options->db && (strcmp(server_options->db, existing->db))) ?
|
|
strdup_root(&mem, server_options->db) : 0;
|
|
|
|
altered->username=
|
|
(server_options->username &&
|
|
(strcmp(server_options->username, existing->username))) ?
|
|
strdup_root(&mem, server_options->username) : 0;
|
|
|
|
altered->password=
|
|
(server_options->password &&
|
|
(strcmp(server_options->password, existing->password))) ?
|
|
strdup_root(&mem, server_options->password) : 0;
|
|
|
|
/*
|
|
port is initialised to -1, so if unset, it will be -1
|
|
*/
|
|
altered->port= (server_options->port > -1 &&
|
|
server_options->port != existing->port) ?
|
|
server_options->port : -1;
|
|
|
|
altered->socket=
|
|
(server_options->socket &&
|
|
(strcmp(server_options->socket, existing->socket))) ?
|
|
strdup_root(&mem, server_options->socket) : 0;
|
|
|
|
altered->scheme=
|
|
(server_options->scheme &&
|
|
(strcmp(server_options->scheme, existing->scheme))) ?
|
|
strdup_root(&mem, server_options->scheme) : 0;
|
|
|
|
altered->owner=
|
|
(server_options->owner &&
|
|
(strcmp(server_options->owner, existing->owner))) ?
|
|
strdup_root(&mem, server_options->owner) : 0;
|
|
|
|
DBUG_VOID_RETURN;
|
|
}
|
|
|
|
/*
|
|
|
|
SYNOPSIS
|
|
servers_free()
|
|
bool end
|
|
|
|
NOTES
|
|
|
|
RETURN VALUE
|
|
void
|
|
|
|
*/
|
|
|
|
void servers_free(bool end)
|
|
{
|
|
DBUG_ENTER("servers_free");
|
|
if (!my_hash_inited(&servers_cache))
|
|
DBUG_VOID_RETURN;
|
|
if (!end)
|
|
{
|
|
free_root(&mem, MYF(MY_MARK_BLOCKS_FREE));
|
|
my_hash_reset(&servers_cache);
|
|
DBUG_VOID_RETURN;
|
|
}
|
|
mysql_rwlock_destroy(&THR_LOCK_servers);
|
|
free_root(&mem,MYF(0));
|
|
my_hash_free(&servers_cache);
|
|
DBUG_VOID_RETURN;
|
|
}
|
|
|
|
|
|
/*
|
|
SYNOPSIS
|
|
|
|
clone_server(MEM_ROOT *mem_root, FOREIGN_SERVER *orig, FOREIGN_SERVER *buff)
|
|
|
|
Create a clone of FOREIGN_SERVER. If the supplied mem_root is of
|
|
thd->mem_root then the copy is automatically disposed at end of statement.
|
|
|
|
NOTES
|
|
|
|
ARGS
|
|
MEM_ROOT pointer (strings are copied into this mem root)
|
|
FOREIGN_SERVER pointer (made a copy of)
|
|
FOREIGN_SERVER buffer (if not-NULL, this pointer is returned)
|
|
|
|
RETURN VALUE
|
|
FOREIGN_SEVER pointer (copy of one supplied FOREIGN_SERVER)
|
|
*/
|
|
|
|
static FOREIGN_SERVER *clone_server(MEM_ROOT *mem, const FOREIGN_SERVER *server,
|
|
FOREIGN_SERVER *buffer)
|
|
{
|
|
DBUG_ENTER("sql_server.cc:clone_server");
|
|
|
|
if (!buffer)
|
|
buffer= (FOREIGN_SERVER *) alloc_root(mem, sizeof(FOREIGN_SERVER));
|
|
|
|
buffer->server_name= strmake_root(mem, server->server_name,
|
|
server->server_name_length);
|
|
buffer->port= server->port;
|
|
buffer->server_name_length= server->server_name_length;
|
|
|
|
/* TODO: We need to examine which of these can really be NULL */
|
|
buffer->db= server->db ? strdup_root(mem, server->db) : NULL;
|
|
buffer->scheme= server->scheme ? strdup_root(mem, server->scheme) : NULL;
|
|
buffer->username= server->username? strdup_root(mem, server->username): NULL;
|
|
buffer->password= server->password? strdup_root(mem, server->password): NULL;
|
|
buffer->socket= server->socket ? strdup_root(mem, server->socket) : NULL;
|
|
buffer->owner= server->owner ? strdup_root(mem, server->owner) : NULL;
|
|
buffer->host= server->host ? strdup_root(mem, server->host) : NULL;
|
|
|
|
DBUG_RETURN(buffer);
|
|
}
|
|
|
|
|
|
/*
|
|
|
|
SYNOPSIS
|
|
get_server_by_name()
|
|
const char *server_name
|
|
|
|
NOTES
|
|
|
|
RETURN VALUE
|
|
FOREIGN_SERVER *
|
|
|
|
*/
|
|
|
|
FOREIGN_SERVER *get_server_by_name(MEM_ROOT *mem, const char *server_name,
|
|
FOREIGN_SERVER *buff)
|
|
{
|
|
size_t server_name_length;
|
|
FOREIGN_SERVER *server;
|
|
DBUG_ENTER("get_server_by_name");
|
|
DBUG_PRINT("info", ("server_name %s", server_name));
|
|
|
|
server_name_length= strlen(server_name);
|
|
|
|
if (! server_name || !strlen(server_name))
|
|
{
|
|
DBUG_PRINT("info", ("server_name not defined!"));
|
|
DBUG_RETURN((FOREIGN_SERVER *)NULL);
|
|
}
|
|
|
|
DBUG_PRINT("info", ("locking servers_cache"));
|
|
mysql_rwlock_rdlock(&THR_LOCK_servers);
|
|
if (!(server= (FOREIGN_SERVER *) my_hash_search(&servers_cache,
|
|
(uchar*) server_name,
|
|
server_name_length)))
|
|
{
|
|
DBUG_PRINT("info", ("server_name %s length %u not found!",
|
|
server_name, (unsigned) server_name_length));
|
|
server= (FOREIGN_SERVER *) NULL;
|
|
}
|
|
/* otherwise, make copy of server */
|
|
else
|
|
server= clone_server(mem, server, buff);
|
|
|
|
DBUG_PRINT("info", ("unlocking servers_cache"));
|
|
mysql_rwlock_unlock(&THR_LOCK_servers);
|
|
DBUG_RETURN(server);
|
|
|
|
}
|