mirror of
https://github.com/MariaDB/server.git
synced 2025-01-31 02:51:44 +01:00
48405ec751
- fix EVENT_ACL problem that GRANT ALL on some_db.* to someone@somewhere did not get to mysql.db - fix crash when the following is executed : CREATE EVENT P() CREATE EVENT E ON SCHEDULER 1 SECOND DO ROLLBACK; (creation works as well as calling P() which creates the event). mysql-test/lib/init_db.sql: - fix init_db.sql so add Event_priv to the database privs, many tests failed because of that ommision - remove the quotes from the column names mysql-test/t/events.test: - fix the small test, don't create own db scripts/mysql_fix_privilege_tables.sql: - fix that sql/event.cc: - be defensive and don't crash if outside has already has opened some table sql/event_executor.cc: - show in SHOW PROCESSLIST - "event_scheduler" as name of the user of the main thread - use "localhost" as the host where event_scheduler comes from - comment out some debug info, fix other debug info sql/event_timed.cc: - enable EVENT creation inside SP. sphead from lex->sphead goes to et->sphead. it's there only if we compile the event. OTOH when doing CREATE PROCEDURE PROC() CREATE EVENT SOME_EV ON SCHEDULE EVERY 1 SECOND DO ROLLBACK; I have only to get the body of the event which is anonymous SP. Before it being "compiled" but then freed without being used because a bit later it is compiled one more time before being put in the events cache. So it was good that the memory structures weren't reused but scrapped out. Now lex->sphead is not needed during event creation but only where the event's body starts and where it ends so to be able at later stage to compile this anonymous SP (the body of the event). sql/sp_head.cc: - copy over a fix to a crash sql/sql_acl.h: - fix privileges. There was _NO_ documentation about that. Another CHUNK had to be created to so EVENT_ACL gets shifted to it's place in the db table. So how this is calculated? EVENT_ACL is 1 << 26. Remember 26, see which poistion in the db table is EVENT_ACL, it's 17, counted from 0. 26 - 17 = 9, then shift it with 9. CHUNKS are created because in some cases some privileges are in chunks and they are shifted at once. There are few chunks of such privileges which has to be shifted to get to exactly the structure of mysql.db table. sql/sql_parse.cc: - ok, we don't care anymore about lex->sphead because our sphead is lex->et->sphead sql/sql_yacc.yy: - bail out if new event_timed returns 0x0 - enable creation of an event inside a SP CREATE PROCEDURE P() CREATE EVENT E ON SCHEDULE EVERY 1 SECOND DO SELECT 1;
970 lines
25 KiB
C++
970 lines
25 KiB
C++
/* Copyright (C) 2004-2005 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 "event_priv.h"
|
|
#include "event.h"
|
|
#include "sp.h"
|
|
|
|
|
|
|
|
extern int yyparse(void *thd);
|
|
|
|
/*
|
|
Init all member variables
|
|
|
|
SYNOPSIS
|
|
event_timed::init()
|
|
*/
|
|
|
|
void
|
|
event_timed::init()
|
|
{
|
|
DBUG_ENTER("event_timed::init");
|
|
|
|
dbname.str= name.str= body.str= comment.str= 0;
|
|
dbname.length= name.length= body.length= comment.length= 0;
|
|
|
|
set_zero_time(&starts, MYSQL_TIMESTAMP_DATETIME);
|
|
set_zero_time(&ends, MYSQL_TIMESTAMP_DATETIME);
|
|
set_zero_time(&execute_at, MYSQL_TIMESTAMP_DATETIME);
|
|
set_zero_time(&last_executed, MYSQL_TIMESTAMP_DATETIME);
|
|
|
|
definer_user.str= definer_host.str= 0;
|
|
definer_user.length= definer_host.length= 0;
|
|
|
|
DBUG_VOID_RETURN;
|
|
}
|
|
|
|
|
|
/*
|
|
Set a name of the event
|
|
|
|
SYNOPSIS
|
|
event_timed::init_name()
|
|
thd THD
|
|
spn the name extracted in the parser
|
|
*/
|
|
|
|
void
|
|
event_timed::init_name(THD *thd, sp_name *spn)
|
|
{
|
|
DBUG_ENTER("event_timed::init_name");
|
|
uint n; /* Counter for nul trimming */
|
|
/* During parsing, we must use thd->mem_root */
|
|
MEM_ROOT *root= thd->mem_root;
|
|
|
|
/* We have to copy strings to get them into the right memroot */
|
|
if (spn)
|
|
{
|
|
dbname.length= spn->m_db.length;
|
|
if (spn->m_db.length == 0)
|
|
dbname.str= NULL;
|
|
else
|
|
dbname.str= strmake_root(root, spn->m_db.str, spn->m_db.length);
|
|
name.length= spn->m_name.length;
|
|
name.str= strmake_root(root, spn->m_name.str, spn->m_name.length);
|
|
|
|
if (spn->m_qname.length == 0)
|
|
spn->init_qname(thd);
|
|
}
|
|
else if (thd->db)
|
|
{
|
|
dbname.length= thd->db_length;
|
|
dbname.str= strmake_root(root, thd->db, dbname.length);
|
|
}
|
|
|
|
DBUG_PRINT("dbname", ("len=%d db=%s",dbname.length, dbname.str));
|
|
DBUG_PRINT("name", ("len=%d name=%s",name.length, name.str));
|
|
|
|
DBUG_VOID_RETURN;
|
|
}
|
|
|
|
|
|
/*
|
|
Set body of the event - what should be executed.
|
|
|
|
SYNOPSIS
|
|
event_timed::init_body()
|
|
thd THD
|
|
|
|
NOTE
|
|
The body is extracted by copying all data between the
|
|
start of the body set by another method and the current pointer in Lex.
|
|
*/
|
|
|
|
void
|
|
event_timed::init_body(THD *thd)
|
|
{
|
|
DBUG_ENTER("event_timed::init_body");
|
|
MEM_ROOT *root= thd->mem_root;
|
|
|
|
body.length= thd->lex->ptr - body_begin;
|
|
// Trim nuls at the end
|
|
while (body.length && body_begin[body.length-1] == '\0')
|
|
body.length--;
|
|
|
|
body.str= strmake_root(root, (char *)body_begin, body.length);
|
|
|
|
DBUG_VOID_RETURN;
|
|
}
|
|
|
|
|
|
/*
|
|
Set time for execution for one time events.
|
|
|
|
SYNOPSIS
|
|
event_timed::init_execute_at()
|
|
expr when (datetime)
|
|
|
|
RETURNS
|
|
0 - OK
|
|
EVEX_PARSE_ERROR - fix_fields failed
|
|
EVEX_BAD_PARAMS - datetime is in the past
|
|
*/
|
|
|
|
int
|
|
event_timed::init_execute_at(THD *thd, Item *expr)
|
|
{
|
|
my_bool not_used;
|
|
TIME ltime;
|
|
my_time_t my_time_tmp;
|
|
|
|
TIME time_tmp;
|
|
DBUG_ENTER("event_timed::init_execute_at");
|
|
|
|
if (expr->fix_fields(thd, &expr))
|
|
DBUG_RETURN(EVEX_PARSE_ERROR);
|
|
|
|
if (expr->val_int() == MYSQL_TIMESTAMP_ERROR)
|
|
DBUG_RETURN(EVEX_BAD_PARAMS);
|
|
|
|
// let's check whether time is in the past
|
|
thd->variables.time_zone->gmt_sec_to_TIME(&time_tmp,
|
|
(my_time_t) thd->query_start());
|
|
|
|
if (expr->val_int() < TIME_to_ulonglong_datetime(&time_tmp))
|
|
DBUG_RETURN(EVEX_BAD_PARAMS);
|
|
|
|
if ((not_used= expr->get_date(<ime, TIME_NO_ZERO_DATE)))
|
|
DBUG_RETURN(EVEX_BAD_PARAMS);
|
|
|
|
/*
|
|
This may result in a 1970-01-01 date if ltime is > 2037-xx-xx
|
|
CONVERT_TZ has similar problem
|
|
*/
|
|
my_tz_UTC->gmt_sec_to_TIME(<ime, TIME_to_timestamp(thd,<ime, ¬_used));
|
|
|
|
|
|
execute_at= ltime;
|
|
DBUG_RETURN(0);
|
|
}
|
|
|
|
|
|
/*
|
|
Set time for execution for transient events.
|
|
|
|
SYNOPSIS
|
|
event_timed::init_interval()
|
|
expr how much?
|
|
new_interval what is the interval
|
|
|
|
RETURNS
|
|
0 - OK
|
|
EVEX_PARSE_ERROR - fix_fields failed
|
|
EVEX_BAD_PARAMS - Interval is not positive
|
|
*/
|
|
|
|
int
|
|
event_timed::init_interval(THD *thd, Item *expr, interval_type new_interval)
|
|
{
|
|
longlong tmp;
|
|
DBUG_ENTER("event_timed::init_interval");
|
|
|
|
if (expr->fix_fields(thd, &expr))
|
|
DBUG_RETURN(EVEX_PARSE_ERROR);
|
|
|
|
if ((tmp= expr->val_int()) <= 0)
|
|
DBUG_RETURN(EVEX_BAD_PARAMS);
|
|
|
|
expression= tmp;
|
|
interval= new_interval;
|
|
DBUG_RETURN(0);
|
|
}
|
|
|
|
|
|
/*
|
|
Set activation time.
|
|
|
|
SYNOPSIS
|
|
event_timed::init_starts()
|
|
expr how much?
|
|
interval what is the interval
|
|
|
|
NOTES
|
|
Note that activation time is not execution time.
|
|
EVERY 5 MINUTE STARTS "2004-12-12 10:00:00" means that
|
|
the event will be executed every 5 minutes but this will
|
|
start at the date shown above. Expressions are possible :
|
|
DATE_ADD(NOW(), INTERVAL 1 DAY) -- start tommorow at
|
|
same time.
|
|
|
|
RETURNS
|
|
0 - OK
|
|
EVEX_PARSE_ERROR - fix_fields failed
|
|
*/
|
|
|
|
int
|
|
event_timed::init_starts(THD *thd, Item *new_starts)
|
|
{
|
|
my_bool not_used;
|
|
TIME ltime;
|
|
my_time_t my_time_tmp;
|
|
|
|
DBUG_ENTER("event_timed::init_starts");
|
|
|
|
if (new_starts->fix_fields(thd, &new_starts))
|
|
DBUG_RETURN(EVEX_PARSE_ERROR);
|
|
|
|
if (new_starts->val_int() == MYSQL_TIMESTAMP_ERROR)
|
|
DBUG_RETURN(EVEX_BAD_PARAMS);
|
|
|
|
if ((not_used= new_starts->get_date(<ime, TIME_NO_ZERO_DATE)))
|
|
DBUG_RETURN(EVEX_BAD_PARAMS);
|
|
|
|
/*
|
|
This may result in a 1970-01-01 date if ltime is > 2037-xx-xx
|
|
CONVERT_TZ has similar problem
|
|
*/
|
|
my_tz_UTC->gmt_sec_to_TIME(<ime, TIME_to_timestamp(thd, <ime, ¬_used));
|
|
|
|
starts= ltime;
|
|
DBUG_RETURN(0);
|
|
}
|
|
|
|
|
|
/*
|
|
Set deactivation time.
|
|
|
|
SYNOPSIS
|
|
event_timed::init_ends()
|
|
thd THD
|
|
new_ends when?
|
|
|
|
NOTES
|
|
Note that activation time is not execution time.
|
|
EVERY 5 MINUTE ENDS "2004-12-12 10:00:00" means that
|
|
the event will be executed every 5 minutes but this will
|
|
end at the date shown above. Expressions are possible :
|
|
DATE_ADD(NOW(), INTERVAL 1 DAY) -- end tommorow at
|
|
same time.
|
|
|
|
RETURNS
|
|
0 - OK
|
|
EVEX_PARSE_ERROR - fix_fields failed
|
|
EVEX_BAD_PARAMS - ENDS before STARTS
|
|
*/
|
|
|
|
int
|
|
event_timed::init_ends(THD *thd, Item *new_ends)
|
|
{
|
|
TIME ltime;
|
|
my_time_t my_time_tmp;
|
|
my_bool not_used;
|
|
|
|
DBUG_ENTER("event_timed::init_ends");
|
|
|
|
if (new_ends->fix_fields(thd, &new_ends))
|
|
DBUG_RETURN(EVEX_PARSE_ERROR);
|
|
|
|
// the field was already fixed in init_ends
|
|
if ((not_used= new_ends->get_date(<ime, TIME_NO_ZERO_DATE)))
|
|
DBUG_RETURN(EVEX_BAD_PARAMS);
|
|
|
|
/*
|
|
This may result in a 1970-01-01 date if ltime is > 2037-xx-xx
|
|
CONVERT_TZ has similar problem
|
|
*/
|
|
my_tz_UTC->gmt_sec_to_TIME(<ime, TIME_to_timestamp(thd, <ime, ¬_used));
|
|
|
|
if (starts.year && my_time_compare(&starts, <ime) != -1)
|
|
DBUG_RETURN(EVEX_BAD_PARAMS);
|
|
|
|
ends= ltime;
|
|
DBUG_RETURN(0);
|
|
}
|
|
|
|
|
|
/*
|
|
Sets comment.
|
|
|
|
SYNOPSIS
|
|
event_timed::init_comment()
|
|
thd THD - used for memory allocation
|
|
comment the string.
|
|
*/
|
|
|
|
void
|
|
event_timed::init_comment(THD *thd, LEX_STRING *set_comment)
|
|
{
|
|
DBUG_ENTER("event_timed::init_comment");
|
|
|
|
comment.str= strmake_root(thd->mem_root, set_comment->str,
|
|
comment.length= set_comment->length);
|
|
|
|
DBUG_VOID_RETURN;
|
|
}
|
|
|
|
|
|
/*
|
|
Inits definer (definer_user and definer_host) during
|
|
parsing.
|
|
|
|
SYNOPSIS
|
|
event_timed::init_definer()
|
|
*/
|
|
|
|
int
|
|
event_timed::init_definer(THD *thd)
|
|
{
|
|
DBUG_ENTER("event_timed::init_definer");
|
|
|
|
definer_user.str= strdup_root(thd->mem_root, thd->security_ctx->priv_user);
|
|
definer_user.length= strlen(thd->security_ctx->priv_user);
|
|
|
|
definer_host.str= strdup_root(thd->mem_root, thd->security_ctx->priv_host);
|
|
definer_host.length= strlen(thd->security_ctx->priv_host);
|
|
|
|
DBUG_RETURN(0);
|
|
}
|
|
|
|
|
|
/*
|
|
Loads an event from a row from mysql.event
|
|
|
|
SYNOPSIS
|
|
event_timed::load_from_row()
|
|
|
|
REMARKS
|
|
This method is silent on errors and should behave like that. Callers
|
|
should handle throwing of error messages. The reason is that the class
|
|
should not know about how to deal with communication.
|
|
*/
|
|
|
|
int
|
|
event_timed::load_from_row(MEM_ROOT *mem_root, TABLE *table)
|
|
{
|
|
longlong created;
|
|
longlong modified;
|
|
char *ptr;
|
|
event_timed *et;
|
|
uint len;
|
|
bool res1, res2;
|
|
|
|
DBUG_ENTER("event_timed::load_from_row");
|
|
|
|
if (!table)
|
|
goto error;
|
|
|
|
et= this;
|
|
|
|
if (table->s->fields != EVEX_FIELD_COUNT)
|
|
goto error;
|
|
|
|
if ((et->dbname.str= get_field(mem_root,
|
|
table->field[EVEX_FIELD_DB])) == NULL)
|
|
goto error;
|
|
|
|
et->dbname.length= strlen(et->dbname.str);
|
|
|
|
if ((et->name.str= get_field(mem_root,
|
|
table->field[EVEX_FIELD_NAME])) == NULL)
|
|
goto error;
|
|
|
|
et->name.length= strlen(et->name.str);
|
|
|
|
if ((et->body.str= get_field(mem_root,
|
|
table->field[EVEX_FIELD_BODY])) == NULL)
|
|
goto error;
|
|
|
|
et->body.length= strlen(et->body.str);
|
|
|
|
if ((et->definer.str= get_field(mem_root,
|
|
table->field[EVEX_FIELD_DEFINER])) == NullS)
|
|
goto error;
|
|
et->definer.length= strlen(et->definer.str);
|
|
|
|
ptr= strchr(et->definer.str, '@');
|
|
|
|
if (! ptr)
|
|
ptr= et->definer.str;
|
|
|
|
len= ptr - et->definer.str;
|
|
|
|
et->definer_user.str= strmake_root(mem_root, et->definer.str, len);
|
|
et->definer_user.length= len;
|
|
len= et->definer.length - len - 1; //1 is because of @
|
|
et->definer_host.str= strmake_root(mem_root, ptr + 1, len);//1: because of @
|
|
et->definer_host.length= len;
|
|
|
|
|
|
res1= table->field[EVEX_FIELD_STARTS]->
|
|
get_date(&et->starts, TIME_NO_ZERO_DATE);
|
|
|
|
res2= table->field[EVEX_FIELD_ENDS]->
|
|
get_date(&et->ends, TIME_NO_ZERO_DATE);
|
|
|
|
et->expression= table->field[EVEX_FIELD_INTERVAL_EXPR]->val_int();
|
|
|
|
/*
|
|
If res1 and res2 are true then both fields are empty.
|
|
Hence if EVEX_FIELD_EXECUTE_AT is empty there is an error.
|
|
*/
|
|
if (res1 && res2 && !et->expression && table->field[EVEX_FIELD_EXECUTE_AT]->
|
|
get_date(&et->execute_at, TIME_NO_ZERO_DATE))
|
|
goto error;
|
|
|
|
/*
|
|
In DB the values start from 1 but enum interval_type starts
|
|
from 0
|
|
*/
|
|
et->interval= (interval_type)
|
|
((ulonglong) table->field[EVEX_FIELD_TRANSIENT_INTERVAL]->val_int() - 1);
|
|
|
|
et->modified= table->field[EVEX_FIELD_CREATED]->val_int();
|
|
et->created= table->field[EVEX_FIELD_MODIFIED]->val_int();
|
|
|
|
/*
|
|
ToDo Andrey : Ask PeterG & Serg what to do in this case.
|
|
Whether on load last_executed_at should be loaded
|
|
or it must be 0ed. If last_executed_at is loaded
|
|
then an event can be scheduled for execution
|
|
instantly. Let's say an event has to be executed
|
|
every 15 mins. The server has been stopped for
|
|
more than this time and then started. If L_E_AT
|
|
is loaded from DB, execution at L_E_AT+15min
|
|
will be scheduled. However this time is in the past.
|
|
Hence immediate execution. Due to patch of
|
|
::mark_last_executed() last_executed gets time_now
|
|
and not execute_at. If not like this a big
|
|
queue can be scheduled for times which are still in
|
|
the past (2, 3 and more executions which will be
|
|
consequent).
|
|
*/
|
|
set_zero_time(&last_executed, MYSQL_TIMESTAMP_DATETIME);
|
|
#ifdef ANDREY_0
|
|
table->field[EVEX_FIELD_LAST_EXECUTED]->
|
|
get_date(&et->last_executed, TIME_NO_ZERO_DATE);
|
|
#endif
|
|
last_executed_changed= false;
|
|
|
|
// ToDo : Andrey . Find a way not to allocate ptr on event_mem_root
|
|
if ((ptr= get_field(mem_root, table->field[EVEX_FIELD_STATUS])) == NullS)
|
|
goto error;
|
|
|
|
DBUG_PRINT("load_from_row", ("Event [%s] is [%s]", et->name.str, ptr));
|
|
et->status= (ptr[0]=='E'? MYSQL_EVENT_ENABLED:
|
|
MYSQL_EVENT_DISABLED);
|
|
|
|
// ToDo : Andrey . Find a way not to allocate ptr on event_mem_root
|
|
if ((ptr= get_field(mem_root,
|
|
table->field[EVEX_FIELD_ON_COMPLETION])) == NullS)
|
|
goto error;
|
|
|
|
et->on_completion= (ptr[0]=='D'? MYSQL_EVENT_ON_COMPLETION_DROP:
|
|
MYSQL_EVENT_ON_COMPLETION_PRESERVE);
|
|
|
|
et->comment.str= get_field(mem_root, table->field[EVEX_FIELD_COMMENT]);
|
|
if (et->comment.str != NullS)
|
|
et->comment.length= strlen(et->comment.str);
|
|
else
|
|
et->comment.length= 0;
|
|
|
|
DBUG_RETURN(0);
|
|
error:
|
|
DBUG_RETURN(EVEX_GET_FIELD_FAILED);
|
|
}
|
|
|
|
|
|
/*
|
|
Note: In the comments this->ends is referenced as m_ends
|
|
|
|
*/
|
|
|
|
bool
|
|
event_timed::compute_next_execution_time()
|
|
{
|
|
TIME time_now;
|
|
my_time_t now;
|
|
int tmp;
|
|
|
|
DBUG_ENTER("event_timed::compute_next_execution_time");
|
|
|
|
if (status == MYSQL_EVENT_DISABLED)
|
|
{
|
|
DBUG_PRINT("compute_next_execution_time",
|
|
("Event %s is DISABLED", name.str));
|
|
goto ret;
|
|
}
|
|
//if one-time no need to do computation
|
|
if (!expression)
|
|
{
|
|
//let's check whether it was executed
|
|
if (last_executed.year)
|
|
{
|
|
DBUG_PRINT("compute_next_execution_time",
|
|
("One-time event %s was already executed", name.str));
|
|
if (on_completion == MYSQL_EVENT_ON_COMPLETION_DROP)
|
|
{
|
|
DBUG_PRINT("compute_next_execution_time",
|
|
("One-time event will be dropped."));
|
|
dropped= true;
|
|
}
|
|
status= MYSQL_EVENT_DISABLED;
|
|
status_changed= true;
|
|
}
|
|
goto ret;
|
|
}
|
|
time(&now);
|
|
my_tz_UTC->gmt_sec_to_TIME(&time_now, now);
|
|
/*
|
|
sql_print_information("[%s.%s]", dbname.str, name.str);
|
|
sql_print_information("time_now : [%d-%d-%d %d:%d:%d ]",
|
|
time_now.year, time_now.month, time_now.day,
|
|
time_now.hour, time_now.minute, time_now.second);
|
|
sql_print_information("starts : [%d-%d-%d %d:%d:%d ]", starts.year,
|
|
starts.month, starts.day, starts.hour,
|
|
starts.minute, starts.second);
|
|
sql_print_information("ends : [%d-%d-%d %d:%d:%d ]", ends.year,
|
|
ends.month, ends.day, ends.hour,
|
|
ends.minute, ends.second);
|
|
sql_print_information("m_last_ex: [%d-%d-%d %d:%d:%d ]", last_executed.year,
|
|
last_executed.month, last_executed.day,
|
|
last_executed.hour, last_executed.minute,
|
|
last_executed.second);
|
|
*/
|
|
//if time_now is after ends don't execute anymore
|
|
if (ends.year && (tmp= my_time_compare(&ends, &time_now)) == -1)
|
|
{
|
|
// time_now is after ends. don't execute anymore
|
|
set_zero_time(&execute_at, MYSQL_TIMESTAMP_DATETIME);
|
|
if (on_completion == MYSQL_EVENT_ON_COMPLETION_DROP)
|
|
dropped= true;
|
|
status= MYSQL_EVENT_DISABLED;
|
|
status_changed= true;
|
|
|
|
goto ret;
|
|
}
|
|
|
|
/*
|
|
Here time_now is before or equals ends if the latter is set.
|
|
Let's check whether time_now is before starts.
|
|
If so schedule for starts
|
|
*/
|
|
if (starts.year && (tmp= my_time_compare(&time_now, &starts)) < 1)
|
|
{
|
|
if (tmp == 0 && my_time_compare(&starts, &last_executed) == 0)
|
|
{
|
|
/*
|
|
time_now = starts = last_executed
|
|
do nothing or we will schedule for second time execution at starts.
|
|
*/
|
|
}
|
|
else
|
|
{
|
|
/*
|
|
starts is in the future
|
|
time_now before starts. Scheduling for starts
|
|
*/
|
|
execute_at= starts;
|
|
goto ret;
|
|
}
|
|
}
|
|
|
|
if (starts.year && ends.year)
|
|
{
|
|
/*
|
|
Both starts and m_ends are set and time_now is between them (incl.)
|
|
If last_executed is set then increase with m_expression. The new TIME is
|
|
after m_ends set execute_at to 0. And check for on_completion
|
|
If not set then schedule for now.
|
|
*/
|
|
if (!last_executed.year)
|
|
execute_at= time_now;
|
|
else
|
|
{
|
|
my_time_t last, ll_ends;
|
|
|
|
// There was previous execution
|
|
last= sec_since_epoch_TIME(&last_executed) + expression;
|
|
ll_ends= sec_since_epoch_TIME(&ends);
|
|
//now convert back to TIME
|
|
//ToDo Andrey: maybe check for error here?
|
|
if (ll_ends < last)
|
|
{
|
|
// Next execution after ends. No more executions
|
|
set_zero_time(&execute_at, MYSQL_TIMESTAMP_DATETIME);
|
|
if (on_completion == MYSQL_EVENT_ON_COMPLETION_DROP)
|
|
dropped= true;
|
|
}
|
|
else
|
|
my_tz_UTC->gmt_sec_to_TIME(&execute_at, last);
|
|
}
|
|
goto ret;
|
|
}
|
|
else if (!starts.year && !ends.year)
|
|
{
|
|
// both starts and m_ends are not set, se we schedule for the next
|
|
// based on last_executed
|
|
if (!last_executed.year)
|
|
//last_executed not set. Schedule the event for now
|
|
execute_at= time_now;
|
|
else
|
|
//ToDo Andrey: maybe check for error here?
|
|
my_tz_UTC->gmt_sec_to_TIME(&execute_at,
|
|
sec_since_epoch_TIME(&last_executed) + expression);
|
|
goto ret;
|
|
}
|
|
else
|
|
{
|
|
//either starts or m_ends is set
|
|
if (starts.year)
|
|
{
|
|
/*
|
|
- starts is set.
|
|
- starts is not in the future according to check made before
|
|
Hence schedule for starts + m_expression in case last_executed
|
|
is not set, otherwise to last_executed + m_expression
|
|
*/
|
|
my_time_t last;
|
|
|
|
//convert either last_executed or starts to seconds
|
|
if (last_executed.year)
|
|
last= sec_since_epoch_TIME(&last_executed) + expression;
|
|
else
|
|
last= sec_since_epoch_TIME(&starts);
|
|
|
|
//now convert back to TIME
|
|
//ToDo Andrey: maybe check for error here?
|
|
my_tz_UTC->gmt_sec_to_TIME(&execute_at, last);
|
|
}
|
|
else
|
|
{
|
|
/*
|
|
- m_ends is set
|
|
- m_ends is after time_now or is equal
|
|
Hence check for m_last_execute and increment with m_expression.
|
|
If last_executed is not set then schedule for now
|
|
*/
|
|
my_time_t last, ll_ends;
|
|
|
|
if (!last_executed.year)
|
|
execute_at= time_now;
|
|
else
|
|
{
|
|
last= sec_since_epoch_TIME(&last_executed);
|
|
ll_ends= sec_since_epoch_TIME(&ends);
|
|
last+= expression;
|
|
//now convert back to TIME
|
|
//ToDo Andrey: maybe check for error here?
|
|
if (ll_ends < last)
|
|
{
|
|
set_zero_time(&execute_at, MYSQL_TIMESTAMP_DATETIME);
|
|
if (on_completion == MYSQL_EVENT_ON_COMPLETION_DROP)
|
|
dropped= true;
|
|
}
|
|
else
|
|
my_tz_UTC->gmt_sec_to_TIME(&execute_at, last);
|
|
}
|
|
}
|
|
goto ret;
|
|
}
|
|
ret:
|
|
|
|
DBUG_RETURN(false);
|
|
}
|
|
|
|
|
|
void
|
|
event_timed::mark_last_executed()
|
|
{
|
|
TIME time_now;
|
|
my_time_t now;
|
|
|
|
time(&now);
|
|
my_tz_UTC->gmt_sec_to_TIME(&time_now, now);
|
|
|
|
last_executed= time_now; // was execute_at
|
|
#ifdef ANDREY_0
|
|
last_executed= execute_at;
|
|
#endif
|
|
last_executed_changed= true;
|
|
}
|
|
|
|
|
|
bool
|
|
event_timed::drop(THD *thd)
|
|
{
|
|
return (bool) evex_drop_event(thd, this, false);
|
|
}
|
|
|
|
|
|
bool
|
|
event_timed::update_fields(THD *thd)
|
|
{
|
|
TABLE *table;
|
|
Open_tables_state backup;
|
|
int ret= 0;
|
|
bool opened;
|
|
|
|
DBUG_ENTER("event_timed::update_time_fields");
|
|
|
|
DBUG_PRINT("enter", ("name: %*s", name.length, name.str));
|
|
|
|
//no need to update if nothing has changed
|
|
if (!(status_changed || last_executed_changed))
|
|
goto done;
|
|
|
|
thd->reset_n_backup_open_tables_state(&backup);
|
|
|
|
if (!(table= evex_open_event_table(thd, TL_WRITE)))
|
|
{
|
|
ret= SP_OPEN_TABLE_FAILED;
|
|
goto done;
|
|
}
|
|
|
|
|
|
if ((ret= evex_db_find_event_aux(thd, dbname, name, table)))
|
|
goto done;
|
|
|
|
store_record(table,record[1]);
|
|
table->timestamp_field_type= TIMESTAMP_NO_AUTO_SET; // Don't update create on row update.
|
|
|
|
if (last_executed_changed)
|
|
{
|
|
table->field[EVEX_FIELD_LAST_EXECUTED]->set_notnull();
|
|
table->field[EVEX_FIELD_LAST_EXECUTED]->store_time(&last_executed,
|
|
MYSQL_TIMESTAMP_DATETIME);
|
|
last_executed_changed= false;
|
|
}
|
|
if (status_changed)
|
|
{
|
|
table->field[EVEX_FIELD_STATUS]->set_notnull();
|
|
table->field[EVEX_FIELD_STATUS]->store((longlong)status);
|
|
status_changed= false;
|
|
}
|
|
|
|
if ((table->file->update_row(table->record[1],table->record[0])))
|
|
ret= EVEX_WRITE_ROW_FAILED;
|
|
|
|
done:
|
|
close_thread_tables(thd);
|
|
thd->restore_backup_open_tables_state(&backup);
|
|
|
|
DBUG_RETURN(ret);
|
|
}
|
|
|
|
|
|
char *
|
|
event_timed::get_show_create_event(THD *thd, uint *length)
|
|
{
|
|
char *dst, *ret;
|
|
uint len, tmp_len;
|
|
|
|
len = strlen("CREATE EVENT ") + dbname.length + strlen(".") + name.length +
|
|
strlen(" ON SCHEDULE EVERY 5 MINUTE DO ") + body.length + strlen(";");
|
|
|
|
ret= dst= (char*) alloc_root(thd->mem_root, len + 1);
|
|
memcpy(dst, "CREATE EVENT ", tmp_len= strlen("CREATE EVENT "));
|
|
dst+= tmp_len;
|
|
memcpy(dst, dbname.str, tmp_len=dbname.length);
|
|
dst+= tmp_len;
|
|
memcpy(dst, ".", tmp_len= strlen("."));
|
|
dst+= tmp_len;
|
|
memcpy(dst, name.str, tmp_len= name.length);
|
|
dst+= tmp_len;
|
|
memcpy(dst, " ON SCHEDULE EVERY 5 MINUTE DO ",
|
|
tmp_len= strlen(" ON SCHEDULE EVERY 5 MINUTE DO "));
|
|
dst+= tmp_len;
|
|
|
|
memcpy(dst, body.str, tmp_len= body.length);
|
|
dst+= tmp_len;
|
|
memcpy(dst, ";", 1);
|
|
++dst;
|
|
*dst= '\0';
|
|
|
|
*length= len;
|
|
|
|
return ret;
|
|
}
|
|
|
|
|
|
/*
|
|
Executes the event (the underlying sp_head object);
|
|
|
|
SYNOPSIS
|
|
evex_fill_row()
|
|
thd THD
|
|
mem_root If != NULL use it to compile the event on it
|
|
|
|
Returns
|
|
0 - success
|
|
-100 - event in execution (parallel execution is impossible)
|
|
others - retcodes of sp_head::execute_procedure()
|
|
|
|
*/
|
|
|
|
int
|
|
event_timed::execute(THD *thd, MEM_ROOT *mem_root)
|
|
{
|
|
List<Item> empty_item_list;
|
|
int ret= 0;
|
|
|
|
DBUG_ENTER("event_timed::execute");
|
|
|
|
VOID(pthread_mutex_lock(&LOCK_running));
|
|
if (running)
|
|
{
|
|
VOID(pthread_mutex_unlock(&LOCK_running));
|
|
DBUG_RETURN(-100);
|
|
}
|
|
running= true;
|
|
VOID(pthread_mutex_unlock(&LOCK_running));
|
|
|
|
// TODO Andrey : make this as member variable and delete in destructor
|
|
empty_item_list.empty();
|
|
|
|
if (!sphead && (ret= compile(thd, mem_root)))
|
|
goto done;
|
|
|
|
ret= sphead->execute_procedure(thd, &empty_item_list);
|
|
|
|
VOID(pthread_mutex_lock(&LOCK_running));
|
|
running= false;
|
|
VOID(pthread_mutex_unlock(&LOCK_running));
|
|
|
|
done:
|
|
// Don't cache sphead if allocated on another mem_root
|
|
if (mem_root && sphead)
|
|
{
|
|
delete sphead;
|
|
sphead= 0;
|
|
}
|
|
|
|
DBUG_RETURN(ret);
|
|
}
|
|
|
|
|
|
/*
|
|
Returns
|
|
0 - Success
|
|
EVEX_COMPILE_ERROR - Error during compilation
|
|
|
|
*/
|
|
|
|
|
|
int
|
|
event_timed::compile(THD *thd, MEM_ROOT *mem_root)
|
|
{
|
|
int ret= 0;
|
|
MEM_ROOT *tmp_mem_root= 0;
|
|
LEX *old_lex= thd->lex, lex;
|
|
char *old_db;
|
|
event_timed *ett;
|
|
sp_name *spn;
|
|
char *old_query;
|
|
uint old_query_len;
|
|
st_sp_chistics *p;
|
|
CHARSET_INFO *old_character_set_client, *old_collation_connection,
|
|
*old_character_set_results;
|
|
|
|
old_character_set_client= thd->variables.character_set_client;
|
|
old_character_set_results= thd->variables.character_set_results;
|
|
old_collation_connection= thd->variables.collation_connection;
|
|
|
|
thd->variables.character_set_client=
|
|
thd->variables.character_set_results=
|
|
thd->variables.collation_connection=
|
|
get_charset_by_csname("utf8", MY_CS_PRIMARY, MYF(MY_WME));
|
|
|
|
thd->update_charset();
|
|
|
|
DBUG_ENTER("event_timed::compile");
|
|
// change the memory root for the execution time
|
|
if (mem_root)
|
|
{
|
|
tmp_mem_root= thd->mem_root;
|
|
thd->mem_root= mem_root;
|
|
}
|
|
old_query_len= thd->query_length;
|
|
old_query= thd->query;
|
|
old_db= thd->db;
|
|
thd->db= dbname.str;
|
|
thd->query= get_show_create_event(thd, &thd->query_length);
|
|
DBUG_PRINT("event_timed::compile", ("query:%s",thd->query));
|
|
|
|
thd->lex= &lex;
|
|
lex_start(thd, (uchar*)thd->query, thd->query_length);
|
|
lex.et_compile_phase= TRUE;
|
|
if (yyparse((void *)thd) || thd->is_fatal_error)
|
|
{
|
|
// Free lex associated resources
|
|
// QQ: Do we really need all this stuff here ?
|
|
if (lex.sphead)
|
|
{
|
|
if (&lex != thd->lex)
|
|
thd->lex->sphead->restore_lex(thd);
|
|
delete lex.sphead;
|
|
lex.sphead= 0;
|
|
}
|
|
// QQ: anything else ?
|
|
lex_end(&lex);
|
|
thd->lex= old_lex;
|
|
|
|
ret= EVEX_COMPILE_ERROR;
|
|
goto done;
|
|
}
|
|
|
|
sphead= lex.et->sphead;
|
|
sphead->m_db= dbname;
|
|
//copy also chistics since they will vanish otherwise we get 0x0 pointer
|
|
// Todo : Handle sql_mode !!
|
|
sphead->set_definer(definer.str, definer.length);
|
|
sphead->set_info(0, 0, &lex.sp_chistics, 0/*sql_mode*/);
|
|
sphead->optimize();
|
|
ret= 0;
|
|
done:
|
|
lex.et->free_sphead_on_delete= false;
|
|
delete lex.et;
|
|
lex_end(&lex);
|
|
thd->lex= old_lex;
|
|
thd->query= old_query;
|
|
thd->query_length= old_query_len;
|
|
thd->db= old_db;
|
|
|
|
thd->variables.character_set_client= old_character_set_client;
|
|
thd->variables.character_set_results= old_character_set_results;
|
|
thd->variables.collation_connection= old_collation_connection;
|
|
thd->update_charset();
|
|
|
|
/*
|
|
Change the memory root for the execution time.
|
|
*/
|
|
if (mem_root)
|
|
thd->mem_root= tmp_mem_root;
|
|
|
|
DBUG_RETURN(ret);
|
|
}
|
|
|