mirror of
https://github.com/MariaDB/server.git
synced 2025-01-31 11:01:52 +01:00
9309fae9b5
5th cut, moved DB related code to Event_db_repository and updated accordingly the remanining code. Moved change/restore_security_context() to class THD Removed events_priv.h Next step is to reorganize create/update_event() and parsing for them. But probably some other refactoring could be done in the meanwhile. The changes so far pass the test suite.
591 lines
15 KiB
C++
591 lines
15 KiB
C++
/* Copyright (C) 2004-2006 MySQL AB
|
|
|
|
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; either version 2 of the License, or
|
|
(at your option) any later version.
|
|
|
|
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 */
|
|
|
|
#include "mysql_priv.h"
|
|
#include "events.h"
|
|
#include "event_data_objects.h"
|
|
#include "event_scheduler.h"
|
|
#include "event_db_repository.h"
|
|
#include "sp_head.h"
|
|
|
|
/*
|
|
TODO list :
|
|
- CREATE EVENT should not go into binary log! Does it now? The SQL statements
|
|
issued by the EVENT are replicated.
|
|
I have an idea how to solve the problem at failover. So the status field
|
|
will be ENUM('DISABLED', 'ENABLED', 'SLAVESIDE_DISABLED').
|
|
In this case when CREATE EVENT is replicated it should go into the binary
|
|
as SLAVESIDE_DISABLED if it is ENABLED, when it's created as DISABLEd it
|
|
should be replicated as disabled. If an event is ALTERed as DISABLED the
|
|
query should go untouched into the binary log, when ALTERed as enable then
|
|
it should go as SLAVESIDE_DISABLED. This is regarding the SQL interface.
|
|
TT routines however modify mysql.event internally and this does not go the
|
|
log so in this case queries has to be injected into the log...somehow... or
|
|
maybe a solution is RBR for this case, because the event may go only from
|
|
ENABLED to DISABLED status change and this is safe for replicating. As well
|
|
an event may be deleted which is also safe for RBR.
|
|
|
|
- Add logging to file
|
|
|
|
Warning:
|
|
- For now parallel execution is not possible because the same sp_head cannot
|
|
be executed few times!!! There is still no lock attached to particular
|
|
event.
|
|
*/
|
|
|
|
|
|
const char *event_scheduler_state_names[]=
|
|
{ "OFF", "0", "ON", "1", "SUSPEND", "2", NullS };
|
|
|
|
TYPELIB Events::opt_typelib=
|
|
{
|
|
array_elements(event_scheduler_state_names)-1,
|
|
"",
|
|
event_scheduler_state_names,
|
|
NULL
|
|
};
|
|
|
|
Events Events::singleton;
|
|
|
|
ulong Events::opt_event_scheduler= 2;
|
|
|
|
|
|
/*
|
|
Compares 2 LEX strings regarding case.
|
|
|
|
SYNOPSIS
|
|
sortcmp_lex_string()
|
|
|
|
s - first LEX_STRING
|
|
t - second LEX_STRING
|
|
cs - charset
|
|
|
|
RETURN VALUE
|
|
-1 - s < t
|
|
0 - s == t
|
|
1 - s > t
|
|
|
|
Notes
|
|
TIME.second_part is not considered during comparison
|
|
*/
|
|
|
|
int sortcmp_lex_string(LEX_STRING s, LEX_STRING t, CHARSET_INFO *cs)
|
|
{
|
|
return cs->coll->strnncollsp(cs, (unsigned char *) s.str,s.length,
|
|
(unsigned char *) t.str,t.length, 0);
|
|
}
|
|
|
|
|
|
Events *
|
|
Events::get_instance()
|
|
{
|
|
DBUG_ENTER("Events::get_instance");
|
|
DBUG_RETURN(&singleton);
|
|
}
|
|
|
|
|
|
/*
|
|
Reconstructs interval expression from interval type and expression
|
|
value that is in form of a value of the smalles entity:
|
|
For
|
|
YEAR_MONTH - expression is in months
|
|
DAY_MINUTE - expression is in minutes
|
|
|
|
SYNOPSIS
|
|
Events::reconstruct_interval_expression()
|
|
buf - preallocated String buffer to add the value to
|
|
interval - the interval type (for instance YEAR_MONTH)
|
|
expression - the value in the lowest entity
|
|
|
|
RETURNS
|
|
0 - OK
|
|
1 - Error
|
|
*/
|
|
|
|
int
|
|
Events::reconstruct_interval_expression(String *buf, interval_type interval,
|
|
longlong expression)
|
|
{
|
|
ulonglong expr= expression;
|
|
char tmp_buff[128], *end;
|
|
bool close_quote= TRUE;
|
|
int multipl= 0;
|
|
char separator=':';
|
|
|
|
switch (interval) {
|
|
case INTERVAL_YEAR_MONTH:
|
|
multipl= 12;
|
|
separator= '-';
|
|
goto common_1_lev_code;
|
|
case INTERVAL_DAY_HOUR:
|
|
multipl= 24;
|
|
separator= ' ';
|
|
goto common_1_lev_code;
|
|
case INTERVAL_HOUR_MINUTE:
|
|
case INTERVAL_MINUTE_SECOND:
|
|
multipl= 60;
|
|
common_1_lev_code:
|
|
buf->append('\'');
|
|
end= longlong10_to_str(expression/multipl, tmp_buff, 10);
|
|
buf->append(tmp_buff, (uint) (end- tmp_buff));
|
|
expr= expr - (expr/multipl)*multipl;
|
|
break;
|
|
case INTERVAL_DAY_MINUTE:
|
|
{
|
|
ulonglong tmp_expr= expr;
|
|
|
|
tmp_expr/=(24*60);
|
|
buf->append('\'');
|
|
end= longlong10_to_str(tmp_expr, tmp_buff, 10);
|
|
buf->append(tmp_buff, (uint) (end- tmp_buff));// days
|
|
buf->append(' ');
|
|
|
|
tmp_expr= expr - tmp_expr*(24*60);//minutes left
|
|
end= longlong10_to_str(tmp_expr/60, tmp_buff, 10);
|
|
buf->append(tmp_buff, (uint) (end- tmp_buff));// hours
|
|
|
|
expr= tmp_expr - (tmp_expr/60)*60;
|
|
/* the code after the switch will finish */
|
|
}
|
|
break;
|
|
case INTERVAL_HOUR_SECOND:
|
|
{
|
|
ulonglong tmp_expr= expr;
|
|
|
|
buf->append('\'');
|
|
end= longlong10_to_str(tmp_expr/3600, tmp_buff, 10);
|
|
buf->append(tmp_buff, (uint) (end- tmp_buff));// hours
|
|
buf->append(':');
|
|
|
|
tmp_expr= tmp_expr - (tmp_expr/3600)*3600;
|
|
end= longlong10_to_str(tmp_expr/60, tmp_buff, 10);
|
|
buf->append(tmp_buff, (uint) (end- tmp_buff));// minutes
|
|
|
|
expr= tmp_expr - (tmp_expr/60)*60;
|
|
/* the code after the switch will finish */
|
|
}
|
|
break;
|
|
case INTERVAL_DAY_SECOND:
|
|
{
|
|
ulonglong tmp_expr= expr;
|
|
|
|
tmp_expr/=(24*3600);
|
|
buf->append('\'');
|
|
end= longlong10_to_str(tmp_expr, tmp_buff, 10);
|
|
buf->append(tmp_buff, (uint) (end- tmp_buff));// days
|
|
buf->append(' ');
|
|
|
|
tmp_expr= expr - tmp_expr*(24*3600);//seconds left
|
|
end= longlong10_to_str(tmp_expr/3600, tmp_buff, 10);
|
|
buf->append(tmp_buff, (uint) (end- tmp_buff));// hours
|
|
buf->append(':');
|
|
|
|
tmp_expr= tmp_expr - (tmp_expr/3600)*3600;
|
|
end= longlong10_to_str(tmp_expr/60, tmp_buff, 10);
|
|
buf->append(tmp_buff, (uint) (end- tmp_buff));// minutes
|
|
|
|
expr= tmp_expr - (tmp_expr/60)*60;
|
|
/* the code after the switch will finish */
|
|
}
|
|
break;
|
|
case INTERVAL_DAY_MICROSECOND:
|
|
case INTERVAL_HOUR_MICROSECOND:
|
|
case INTERVAL_MINUTE_MICROSECOND:
|
|
case INTERVAL_SECOND_MICROSECOND:
|
|
case INTERVAL_MICROSECOND:
|
|
my_error(ER_NOT_SUPPORTED_YET, MYF(0), "MICROSECOND");
|
|
return 1;
|
|
break;
|
|
case INTERVAL_QUARTER:
|
|
expr/= 3;
|
|
close_quote= FALSE;
|
|
break;
|
|
case INTERVAL_WEEK:
|
|
expr/= 7;
|
|
default:
|
|
close_quote= FALSE;
|
|
break;
|
|
}
|
|
if (close_quote)
|
|
buf->append(separator);
|
|
end= longlong10_to_str(expr, tmp_buff, 10);
|
|
buf->append(tmp_buff, (uint) (end- tmp_buff));
|
|
if (close_quote)
|
|
buf->append('\'');
|
|
|
|
return 0;
|
|
}
|
|
|
|
|
|
/*
|
|
Open mysql.event table for read
|
|
|
|
SYNOPSIS
|
|
Events::open_event_table()
|
|
thd Thread context
|
|
lock_type How to lock the table
|
|
table We will store the open table here
|
|
|
|
RETURN VALUE
|
|
1 Cannot lock table
|
|
2 The table is corrupted - different number of fields
|
|
0 OK
|
|
*/
|
|
|
|
int
|
|
Events::open_event_table(THD *thd, enum thr_lock_type lock_type,
|
|
TABLE **table)
|
|
{
|
|
return db_repository.open_event_table(thd, lock_type, table);
|
|
}
|
|
|
|
|
|
/*
|
|
The function exported to the world for creating of events.
|
|
|
|
SYNOPSIS
|
|
Events::create_event()
|
|
thd THD
|
|
et event's data
|
|
create_options Options specified when in the query. We are
|
|
interested whether there is IF NOT EXISTS
|
|
rows_affected How many rows were affected
|
|
|
|
RETURN VALUE
|
|
0 OK
|
|
!0 Error
|
|
|
|
NOTES
|
|
- in case there is an event with the same name (db) and
|
|
IF NOT EXISTS is specified, an warning is put into the W stack.
|
|
*/
|
|
|
|
int
|
|
Events::create_event(THD *thd, Event_timed *et, Event_parse_data *parse_data,
|
|
uint create_options, uint *rows_affected)
|
|
{
|
|
int ret;
|
|
|
|
DBUG_ENTER("Events::create_event");
|
|
DBUG_PRINT("enter", ("name: %*s options:%d", et->name.length,
|
|
et->name.str, create_options));
|
|
|
|
if (!(ret= db_repository.
|
|
create_event(thd, et,
|
|
create_options & HA_LEX_CREATE_IF_NOT_EXISTS,
|
|
rows_affected)))
|
|
{
|
|
Event_scheduler *scheduler= Event_scheduler::get_instance();
|
|
if (scheduler->initialized() &&
|
|
(ret= scheduler->create_event(thd, et, true)))
|
|
my_error(ER_EVENT_MODIFY_QUEUE_ERROR, MYF(0), ret);
|
|
}
|
|
/* No need to close the table, it will be closed in sql_parse::do_command */
|
|
|
|
DBUG_RETURN(ret);
|
|
}
|
|
|
|
|
|
/*
|
|
The function exported to the world for alteration of events.
|
|
|
|
SYNOPSIS
|
|
Events::update_event()
|
|
thd THD
|
|
et event's data
|
|
new_name set in case of RENAME TO.
|
|
|
|
RETURN VALUE
|
|
0 OK
|
|
!0 Error
|
|
|
|
NOTES
|
|
et contains data about dbname and event name.
|
|
new_name is the new name of the event, if not null (this means
|
|
that RENAME TO was specified in the query)
|
|
*/
|
|
|
|
int
|
|
Events::update_event(THD *thd, Event_timed *et, Event_parse_data *parse_data,
|
|
sp_name *new_name, uint *rows_affected)
|
|
{
|
|
int ret;
|
|
|
|
DBUG_ENTER("Events::update_event");
|
|
DBUG_PRINT("enter", ("name: %*s", et->name.length, et->name.str));
|
|
/*
|
|
db_update_event() opens & closes the table to prevent
|
|
crash later in the code when loading and compiling the new definition.
|
|
Also on error conditions my_error() is called so no need to handle here
|
|
*/
|
|
if (!(ret= db_repository.update_event(thd, et, new_name)))
|
|
{
|
|
Event_scheduler *scheduler= Event_scheduler::get_instance();
|
|
if (scheduler->initialized() &&
|
|
(ret= scheduler->update_event(thd, et,
|
|
new_name? &new_name->m_db: NULL,
|
|
new_name? &new_name->m_name: NULL)))
|
|
my_error(ER_EVENT_MODIFY_QUEUE_ERROR, MYF(0), ret);
|
|
}
|
|
DBUG_RETURN(ret);
|
|
}
|
|
|
|
|
|
/*
|
|
Drops an event
|
|
|
|
SYNOPSIS
|
|
Events::drop_event()
|
|
thd THD
|
|
name event's name
|
|
drop_if_exists if set and the event not existing => warning onto the stack
|
|
rows_affected affected number of rows is returned heres
|
|
|
|
RETURN VALUE
|
|
0 OK
|
|
!0 Error (reported)
|
|
*/
|
|
|
|
int
|
|
Events::drop_event(THD *thd, sp_name *name, bool drop_if_exists,
|
|
uint *rows_affected)
|
|
{
|
|
int ret;
|
|
|
|
DBUG_ENTER("Events::drop_event");
|
|
|
|
if (!(ret= db_repository.drop_event(thd, name->m_db, name->m_name,
|
|
drop_if_exists, rows_affected)))
|
|
{
|
|
Event_scheduler *scheduler= Event_scheduler::get_instance();
|
|
if (scheduler->initialized() && (ret= scheduler->drop_event(thd, name)))
|
|
my_error(ER_EVENT_MODIFY_QUEUE_ERROR, MYF(0), ret);
|
|
}
|
|
DBUG_RETURN(ret);
|
|
}
|
|
|
|
|
|
/*
|
|
SHOW CREATE EVENT
|
|
|
|
SYNOPSIS
|
|
Events::show_create_event()
|
|
thd THD
|
|
spn the name of the event (db, name)
|
|
|
|
RETURN VALUE
|
|
0 OK
|
|
1 Error during writing to the wire
|
|
*/
|
|
|
|
int
|
|
Events::show_create_event(THD *thd, sp_name *spn)
|
|
{
|
|
int ret;
|
|
Event_timed *et= NULL;
|
|
Open_tables_state backup;
|
|
|
|
DBUG_ENTER("Events::show_create_event");
|
|
DBUG_PRINT("enter", ("name: %*s", spn->m_name.length, spn->m_name.str));
|
|
|
|
thd->reset_n_backup_open_tables_state(&backup);
|
|
ret= db_repository.find_event(thd, spn, &et, NULL, thd->mem_root);
|
|
thd->restore_backup_open_tables_state(&backup);
|
|
|
|
if (!ret)
|
|
{
|
|
Protocol *protocol= thd->protocol;
|
|
char show_str_buf[768];
|
|
String show_str(show_str_buf, sizeof(show_str_buf), system_charset_info);
|
|
List<Item> field_list;
|
|
byte *sql_mode_str;
|
|
ulong sql_mode_len=0;
|
|
|
|
show_str.length(0);
|
|
show_str.set_charset(system_charset_info);
|
|
|
|
if (et->get_create_event(thd, &show_str))
|
|
goto err;
|
|
|
|
field_list.push_back(new Item_empty_string("Event", NAME_LEN));
|
|
|
|
sql_mode_str=
|
|
sys_var_thd_sql_mode::symbolic_mode_representation(thd, et->sql_mode,
|
|
&sql_mode_len);
|
|
|
|
field_list.push_back(new Item_empty_string("sql_mode", sql_mode_len));
|
|
|
|
field_list.push_back(new Item_empty_string("Create Event",
|
|
show_str.length()));
|
|
if (protocol->send_fields(&field_list, Protocol::SEND_NUM_ROWS |
|
|
Protocol::SEND_EOF))
|
|
goto err;
|
|
|
|
protocol->prepare_for_resend();
|
|
protocol->store(et->name.str, et->name.length, system_charset_info);
|
|
|
|
protocol->store((char*) sql_mode_str, sql_mode_len, system_charset_info);
|
|
|
|
protocol->store(show_str.c_ptr(), show_str.length(), system_charset_info);
|
|
ret= protocol->write();
|
|
send_eof(thd);
|
|
}
|
|
delete et;
|
|
DBUG_RETURN(ret);
|
|
err:
|
|
delete et;
|
|
DBUG_RETURN(1);
|
|
}
|
|
|
|
|
|
/*
|
|
Drops all events from a schema
|
|
|
|
SYNOPSIS
|
|
Events::drop_schema_events()
|
|
thd Thread
|
|
db ASCIIZ schema name
|
|
|
|
RETURN VALUE
|
|
0 OK
|
|
!0 Error
|
|
*/
|
|
|
|
int
|
|
Events::drop_schema_events(THD *thd, char *db)
|
|
{
|
|
int ret= 0;
|
|
LEX_STRING db_lex= {db, strlen(db)};
|
|
|
|
DBUG_ENTER("evex_drop_db_events");
|
|
DBUG_PRINT("enter", ("dropping events from %s", db));
|
|
|
|
Event_scheduler *scheduler= Event_scheduler::get_instance();
|
|
ret= scheduler->drop_schema_events(thd, db_lex);
|
|
ret= db_repository.drop_schema_events(thd, db_lex);
|
|
|
|
DBUG_RETURN(ret);
|
|
}
|
|
|
|
|
|
/*
|
|
Inits the scheduler's structures.
|
|
|
|
SYNOPSIS
|
|
Events::init()
|
|
|
|
NOTES
|
|
This function is not synchronized.
|
|
|
|
RETURN VALUE
|
|
0 OK
|
|
1 Error
|
|
*/
|
|
|
|
int
|
|
Events::init()
|
|
{
|
|
int ret= 0;
|
|
Event_db_repository *db_repo;
|
|
DBUG_ENTER("Events::init");
|
|
db_repo= &get_instance()->db_repository;
|
|
db_repo->init_repository();
|
|
|
|
/* it should be an assignment! */
|
|
if (opt_event_scheduler)
|
|
{
|
|
Event_scheduler *scheduler= Event_scheduler::get_instance();
|
|
DBUG_ASSERT(opt_event_scheduler == 1 || opt_event_scheduler == 2);
|
|
DBUG_RETURN(scheduler->init(db_repo) ||
|
|
(opt_event_scheduler == 1? scheduler->start():
|
|
scheduler->start_suspended()));
|
|
}
|
|
DBUG_RETURN(0);
|
|
}
|
|
|
|
|
|
/*
|
|
Cleans up scheduler's resources. Called at server shutdown.
|
|
|
|
SYNOPSIS
|
|
Events::deinit()
|
|
|
|
NOTES
|
|
This function is not synchronized.
|
|
*/
|
|
|
|
void
|
|
Events::deinit()
|
|
{
|
|
DBUG_ENTER("Events::deinit");
|
|
Event_scheduler *scheduler= Event_scheduler::get_instance();
|
|
if (scheduler->initialized())
|
|
{
|
|
scheduler->stop();
|
|
scheduler->destroy();
|
|
}
|
|
get_instance()->db_repository.deinit_repository();
|
|
|
|
DBUG_VOID_RETURN;
|
|
}
|
|
|
|
|
|
/*
|
|
Proxy for Event_scheduler::dump_internal_status
|
|
|
|
SYNOPSIS
|
|
Events::dump_internal_status()
|
|
thd Thread
|
|
|
|
RETURN VALUE
|
|
0 OK
|
|
!0 Error
|
|
*/
|
|
|
|
int
|
|
Events::dump_internal_status(THD *thd)
|
|
{
|
|
return Event_scheduler::dump_internal_status(thd);
|
|
}
|
|
|
|
|
|
/*
|
|
Inits Events mutexes
|
|
|
|
SYNOPSIS
|
|
Events::init_mutexes()
|
|
thd Thread
|
|
*/
|
|
|
|
void
|
|
Events::init_mutexes()
|
|
{
|
|
Event_scheduler::init_mutexes();
|
|
}
|
|
|
|
|
|
/*
|
|
Destroys Events mutexes
|
|
|
|
SYNOPSIS
|
|
Events::destroy_mutexes()
|
|
*/
|
|
|
|
void
|
|
Events::destroy_mutexes()
|
|
{
|
|
Event_scheduler::destroy_mutexes();
|
|
}
|