2006-06-27 08:48:50 +02:00
|
|
|
#ifndef _EVENT_DATA_OBJECTS_H_
|
|
|
|
#define _EVENT_DATA_OBJECTS_H_
|
2006-06-20 17:05:41 +02:00
|
|
|
/* 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 */
|
|
|
|
|
|
|
|
|
2006-07-11 18:28:15 +02:00
|
|
|
#define EVEX_GET_FIELD_FAILED -2
|
|
|
|
#define EVEX_COMPILE_ERROR -3
|
|
|
|
#define EVEX_GENERAL_ERROR -4
|
|
|
|
#define EVEX_BAD_PARAMS -5
|
|
|
|
#define EVEX_MICROSECOND_UNSUP -6
|
2006-06-28 01:28:03 +02:00
|
|
|
|
|
|
|
|
2006-06-20 17:05:41 +02:00
|
|
|
class sp_head;
|
2006-06-27 08:48:50 +02:00
|
|
|
class Sql_alloc;
|
2006-06-28 01:28:03 +02:00
|
|
|
|
|
|
|
|
2006-07-10 13:44:43 +02:00
|
|
|
class Event_basic
|
2006-06-20 17:05:41 +02:00
|
|
|
{
|
2006-07-10 13:44:43 +02:00
|
|
|
protected:
|
|
|
|
MEM_ROOT mem_root;
|
2006-06-20 17:05:41 +02:00
|
|
|
|
2006-07-10 13:44:43 +02:00
|
|
|
public:
|
|
|
|
LEX_STRING dbname;
|
|
|
|
LEX_STRING name;
|
|
|
|
LEX_STRING definer;// combination of user and host
|
|
|
|
|
|
|
|
Event_basic();
|
|
|
|
virtual ~Event_basic();
|
|
|
|
|
|
|
|
virtual int
|
|
|
|
load_from_row(TABLE *table) = 0;
|
|
|
|
|
|
|
|
protected:
|
|
|
|
bool
|
|
|
|
load_string_fields(Field **fields, ...);
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
class Event_queue_element : public Event_basic
|
|
|
|
{
|
|
|
|
protected:
|
2006-06-20 17:05:41 +02:00
|
|
|
bool status_changed;
|
|
|
|
bool last_executed_changed;
|
|
|
|
|
|
|
|
public:
|
|
|
|
enum enum_status
|
|
|
|
{
|
|
|
|
ENABLED = 1,
|
|
|
|
DISABLED
|
|
|
|
};
|
|
|
|
|
|
|
|
enum enum_on_completion
|
|
|
|
{
|
|
|
|
ON_COMPLETION_DROP = 1,
|
|
|
|
ON_COMPLETION_PRESERVE
|
|
|
|
};
|
|
|
|
|
2006-07-10 13:44:43 +02:00
|
|
|
enum enum_on_completion on_completion;
|
|
|
|
enum enum_status status;
|
2006-06-20 17:05:41 +02:00
|
|
|
TIME last_executed;
|
|
|
|
|
|
|
|
TIME starts;
|
|
|
|
TIME ends;
|
|
|
|
TIME execute_at;
|
|
|
|
my_bool starts_null;
|
|
|
|
my_bool ends_null;
|
|
|
|
my_bool execute_at_null;
|
|
|
|
|
|
|
|
longlong expression;
|
|
|
|
interval_type interval;
|
|
|
|
|
2006-07-10 13:44:43 +02:00
|
|
|
uint flags;//all kind of purposes
|
2006-06-20 17:05:41 +02:00
|
|
|
|
|
|
|
bool dropped;
|
2006-07-10 13:44:43 +02:00
|
|
|
|
|
|
|
Event_queue_element();
|
|
|
|
virtual ~Event_queue_element();
|
|
|
|
|
|
|
|
virtual int
|
|
|
|
load_from_row(TABLE *table);
|
|
|
|
|
|
|
|
bool
|
|
|
|
compute_next_execution_time();
|
|
|
|
|
|
|
|
int
|
|
|
|
drop(THD *thd);
|
|
|
|
|
|
|
|
void
|
|
|
|
mark_last_executed(THD *thd);
|
|
|
|
|
|
|
|
bool
|
|
|
|
update_timing_fields(THD *thd);
|
2006-06-20 17:05:41 +02:00
|
|
|
|
|
|
|
static void *operator new(size_t size)
|
|
|
|
{
|
|
|
|
void *p;
|
2006-07-10 13:44:43 +02:00
|
|
|
DBUG_ENTER("Event_queue_element::new(size)");
|
2006-06-20 17:05:41 +02:00
|
|
|
p= my_malloc(size, MYF(0));
|
|
|
|
DBUG_PRINT("info", ("alloc_ptr=0x%lx", p));
|
|
|
|
DBUG_RETURN(p);
|
|
|
|
}
|
|
|
|
|
|
|
|
static void operator delete(void *ptr, size_t size)
|
|
|
|
{
|
2006-07-10 13:44:43 +02:00
|
|
|
DBUG_ENTER("Event_queue_element::delete(ptr,size)");
|
2006-06-20 17:05:41 +02:00
|
|
|
DBUG_PRINT("enter", ("free_ptr=0x%lx", ptr));
|
|
|
|
TRASH(ptr, size);
|
|
|
|
my_free((gptr) ptr, MYF(0));
|
|
|
|
DBUG_VOID_RETURN;
|
|
|
|
}
|
2006-07-10 13:44:43 +02:00
|
|
|
};
|
2006-06-20 17:05:41 +02:00
|
|
|
|
|
|
|
|
2006-07-10 13:44:43 +02:00
|
|
|
class Event_timed : public Event_queue_element
|
|
|
|
{
|
|
|
|
Event_timed(const Event_timed &); /* Prevent use of these */
|
|
|
|
void operator=(Event_timed &);
|
2006-06-20 17:05:41 +02:00
|
|
|
|
2006-07-10 13:44:43 +02:00
|
|
|
public:
|
|
|
|
LEX_STRING body;
|
2006-06-20 17:05:41 +02:00
|
|
|
|
2006-07-10 13:44:43 +02:00
|
|
|
LEX_STRING definer_user;
|
|
|
|
LEX_STRING definer_host;
|
2006-06-20 17:05:41 +02:00
|
|
|
|
2006-07-10 13:44:43 +02:00
|
|
|
LEX_STRING comment;
|
2006-06-20 17:05:41 +02:00
|
|
|
|
2006-07-10 13:44:43 +02:00
|
|
|
ulonglong created;
|
|
|
|
ulonglong modified;
|
|
|
|
|
|
|
|
ulong sql_mode;
|
|
|
|
|
|
|
|
Event_timed();
|
|
|
|
virtual ~Event_timed();
|
2006-06-20 17:05:41 +02:00
|
|
|
|
|
|
|
void
|
2006-07-10 13:44:43 +02:00
|
|
|
init();
|
2006-06-20 17:05:41 +02:00
|
|
|
|
2006-07-10 13:44:43 +02:00
|
|
|
virtual int
|
|
|
|
load_from_row(TABLE *table);
|
2006-06-20 17:05:41 +02:00
|
|
|
|
|
|
|
int
|
|
|
|
get_create_event(THD *thd, String *buf);
|
2006-07-10 13:44:43 +02:00
|
|
|
};
|
|
|
|
|
|
|
|
|
|
|
|
class Event_job_data : public Event_basic
|
|
|
|
{
|
|
|
|
public:
|
|
|
|
THD *thd;
|
|
|
|
sp_head *sphead;
|
|
|
|
|
|
|
|
LEX_STRING body;
|
|
|
|
LEX_STRING definer_user;
|
|
|
|
LEX_STRING definer_host;
|
|
|
|
|
|
|
|
ulong sql_mode;
|
|
|
|
|
|
|
|
Event_job_data();
|
|
|
|
virtual ~Event_job_data();
|
|
|
|
|
|
|
|
virtual int
|
|
|
|
load_from_row(TABLE *table);
|
2006-06-20 17:05:41 +02:00
|
|
|
|
|
|
|
int
|
2006-07-11 18:28:15 +02:00
|
|
|
execute(THD *thd);
|
2006-07-10 13:44:43 +02:00
|
|
|
private:
|
|
|
|
int
|
|
|
|
get_fake_create_event(THD *thd, String *buf);
|
2006-06-20 17:05:41 +02:00
|
|
|
|
|
|
|
int
|
|
|
|
compile(THD *thd, MEM_ROOT *mem_root);
|
2006-07-10 13:44:43 +02:00
|
|
|
|
2006-06-20 17:05:41 +02:00
|
|
|
void
|
|
|
|
free_sp();
|
2006-07-10 13:44:43 +02:00
|
|
|
|
|
|
|
Event_job_data(const Event_job_data &); /* Prevent use of these */
|
|
|
|
void operator=(Event_job_data &);
|
2006-06-20 17:05:41 +02:00
|
|
|
};
|
2006-06-27 08:48:50 +02:00
|
|
|
|
|
|
|
|
|
|
|
class Event_parse_data : public Sql_alloc
|
|
|
|
{
|
|
|
|
public:
|
|
|
|
enum enum_status
|
|
|
|
{
|
|
|
|
ENABLED = 1,
|
|
|
|
DISABLED
|
|
|
|
};
|
|
|
|
|
|
|
|
enum enum_on_completion
|
|
|
|
{
|
|
|
|
ON_COMPLETION_DROP = 1,
|
|
|
|
ON_COMPLETION_PRESERVE
|
|
|
|
};
|
|
|
|
enum enum_on_completion on_completion;
|
|
|
|
enum enum_status status;
|
|
|
|
|
|
|
|
const uchar *body_begin;
|
|
|
|
|
|
|
|
LEX_STRING dbname;
|
|
|
|
LEX_STRING name;
|
|
|
|
LEX_STRING definer;// combination of user and host
|
2006-07-10 13:44:43 +02:00
|
|
|
LEX_STRING body;
|
2006-06-27 08:48:50 +02:00
|
|
|
LEX_STRING comment;
|
WL#3337 (Events new architecture)
This cut No 7 should finish the part of fixing the parsing of the events :
- Event_timed is no more used during parsing. Less problems because it has
a mutex. Event_parse_data class is used during parsing. It is suited only
for this purpose. It's pretty lightweight
- Late checking of data from parsing is being performed. This should solve
the problems of nested events in SP or other events (for the situation
of no nested bodies). Before if an ALTER EVENT was in a SP, then when the
SP was compiled, and not executed, the actual init_xxx methods of Event_timed
were called, which is wrong.
- It could be a side effect of using a specialized class, but test events_stress is
now 25% quicker.
Cut No8 will start splitting Event_scheduler into 2 parts, the QUEUE will be moved
to Event_queue.
mysql-test/r/events.result:
update result
mysql-test/t/events.test:
disabled is actually wrong, should be disable, but because of the early
checking it was never parsed.
sql/event_data_objects.cc:
move add init_xxx methods from Event_timed to Event_parse_data
Event_parse data does not need definer_user and definer_host
in Event_timed::compile() do not use lex.et, well there is no more lex.et :)
sql/event_data_objects.h:
move parsing responsibilities from Event_timed to Event_parse_data
sql/event_db_repository.cc:
No more Event_timed comes from parsing but Event_parse_data
The initialization of Item*-s from parsing is done late, and not
during the actual parsing. This is the right way to go because
if an ALTER EVENT is inside a SP or CREATE EVENT it should not be
executed (initialized) during parsing, as it was done.
sql/event_db_repository.h:
No more Event_timed comes from parsing but Event_parse_data
The initialization of Item*-s from parsing is done late, and not
during the actual parsing. This is the right way to go because
if an ALTER EVENT is inside a SP or CREATE EVENT it should not be
executed (initialized) during parsing, as it was done.
sql/event_scheduler.cc:
No more Event_timed comes from parsing but Event_parse_data
The initialization of Item*-s from parsing is done late, and not
during the actual parsing. This is the right way to go because
if an ALTER EVENT is inside a SP or CREATE EVENT it should not be
executed (initialized) during parsing, as it was done.
sql/event_scheduler.h:
No more Event_timed comes from parsing but Event_parse_data
The initialization of Item*-s from parsing is done late, and not
during the actual parsing. This is the right way to go because
if an ALTER EVENT is inside a SP or CREATE EVENT it should not be
executed (initialized) during parsing, as it was done.
sql/events.cc:
No more Event_timed comes from parsing but Event_parse_data
The initialization of Item*-s from parsing is done late, and not
during the actual parsing. This is the right way to go because
if an ALTER EVENT is inside a SP or CREATE EVENT it should not be
executed (initialized) during parsing, as it was done.
sql/events.h:
No more Event_timed comes from parsing but Event_parse_data
The initialization of Item*-s from parsing is done late, and not
during the actual parsing. This is the right way to go because
if an ALTER EVENT is inside a SP or CREATE EVENT it should not be
executed (initialized) during parsing, as it was done.
sql/sql_lex.cc:
lex->et_compile_phase and lex->et are no more.
Use lex->event_parse_data
sql/sql_lex.h:
lex->et_compile_phase and lex->et are no more.
Use lex->event_parse_data
sql/sql_parse.cc:
lex->et_compile_phase and lex->et are no more.
Use lex->event_parse_data
ACL checks were moved inside the Events subsystem.
Also ending of the transaction is performed only just
before doing disk operation. Therefore only when needed.
sql/sql_yacc.yy:
lex->et and lex->et_parse_phase are no more
Use the specialized for parsing Event_parse_data
2006-06-29 00:42:25 +02:00
|
|
|
|
2006-06-27 08:48:50 +02:00
|
|
|
Item* item_starts;
|
|
|
|
Item* item_ends;
|
|
|
|
Item* item_execute_at;
|
|
|
|
|
|
|
|
TIME starts;
|
|
|
|
TIME ends;
|
|
|
|
TIME execute_at;
|
|
|
|
my_bool starts_null;
|
|
|
|
my_bool ends_null;
|
|
|
|
my_bool execute_at_null;
|
|
|
|
|
WL#3337 (Events new infrasctructure)
Second cut of separating parsing phase from execution phase
Separate Event_timed from parsing phase and introducing Event_parse_data.
sql/event_data_objects.cc:
second cut,
copy Event_timed::init_body() to Event_parse_data::init_body()
Init the body during parsing, everything else keep as a pointer to
Item or some other pointer to use for later initialization.
sql/event_data_objects.h:
get the identifier as sp_name*, later will initialize our structures
sql/events.cc:
for easy transition add temporarily parse_data, later Event_timed *et will be removed.
Do slow transition because Event_timed is so tightly integrated that a front-attack
by removing things from this class was unsuccessful. Do things step by step by eliminating
dependencies. Hence, the code could be checked with the current test suite too
(early testing)
sql/events.h:
for easy transition add temporarily parse_data, later Event_timed *et will be removed.
Do slow transition because Event_timed is so tightly integrated that a front-attack
by removing things from this class was unsuccessful. Do things step by step by eliminating
dependencies. Hence, the code could be checked with the current test suite too
(early testing)
BitKeeper/etc/ignore:
Added libmysql/viosocket.o.6WmSJk libmysqld/event_data_objects.cc libmysqld/event_db_repository.cc libmysqld/event_queue.cc server-tools/instance-manager/net_serv.cc to the ignore list
sql/share/errmsg.txt:
remove this message, not used and needed for now
sql/sql_lex.h:
for easy transition add temporarily parse_data, later Event_timed *et will be removed.
Do slow transition because Event_timed is so tightly integrated that a front-attack
by removing things from this class was unsuccessful. Do things step by step by eliminating
dependencies. Hence, the code could be checked with the current test suite too
(early testing)
sql/sql_parse.cc:
for easy transition add temporarily parse_data, later Event_timed *et will be removed.
Do slow transition because Event_timed is so tightly integrated that a front-attack
by removing things from this class was unsuccessful. Do things step by step by eliminating
dependencies. Hence, the code could be checked with the current test suite too
(early testing)
sql/sql_yacc.yy:
for easy transition add temporarily parse_data, later Event_timed *et will be removed.
Do slow transition because Event_timed is so tightly integrated that a front-attack
by removing things from this class was unsuccessful. Do things step by step by eliminating
dependencies. Hence, the code could be checked with the current test suite too
(early testing)
2006-06-27 10:53:26 +02:00
|
|
|
sp_name *identifier;
|
2006-06-27 08:48:50 +02:00
|
|
|
Item* item_expression;
|
|
|
|
longlong expression;
|
|
|
|
interval_type interval;
|
|
|
|
|
|
|
|
static Event_parse_data *
|
|
|
|
new_instance(THD *thd);
|
|
|
|
|
2006-07-10 13:44:43 +02:00
|
|
|
bool
|
|
|
|
check_parse_data(THD *);
|
2006-06-27 08:48:50 +02:00
|
|
|
|
2006-07-10 13:44:43 +02:00
|
|
|
void
|
|
|
|
init_body(THD *thd);
|
2006-06-27 08:48:50 +02:00
|
|
|
|
2006-07-10 13:44:43 +02:00
|
|
|
private:
|
2006-06-27 08:48:50 +02:00
|
|
|
|
2006-07-11 18:28:15 +02:00
|
|
|
void
|
2006-07-10 13:44:43 +02:00
|
|
|
init_definer(THD *thd);
|
2006-06-27 08:48:50 +02:00
|
|
|
|
|
|
|
void
|
|
|
|
init_name(THD *thd, sp_name *spn);
|
|
|
|
|
|
|
|
int
|
2006-07-10 13:44:43 +02:00
|
|
|
init_execute_at(THD *thd);
|
2006-06-27 08:48:50 +02:00
|
|
|
|
|
|
|
int
|
2006-07-10 13:44:43 +02:00
|
|
|
init_interval(THD *thd);
|
2006-06-28 01:28:03 +02:00
|
|
|
|
2006-07-04 18:44:35 +02:00
|
|
|
int
|
2006-07-10 13:44:43 +02:00
|
|
|
init_starts(THD *thd);
|
2006-07-04 18:44:35 +02:00
|
|
|
|
|
|
|
int
|
2006-07-10 13:44:43 +02:00
|
|
|
init_ends(THD *thd);
|
|
|
|
|
|
|
|
Event_parse_data();
|
|
|
|
~Event_parse_data();
|
2006-07-04 18:44:35 +02:00
|
|
|
|
2006-07-10 13:44:43 +02:00
|
|
|
void
|
|
|
|
report_bad_value(const char *item_name, Item *bad_item);
|
2006-07-04 18:44:35 +02:00
|
|
|
|
2006-07-10 13:44:43 +02:00
|
|
|
Event_parse_data(const Event_parse_data &); /* Prevent use of these */
|
|
|
|
void operator=(Event_parse_data &);
|
2006-07-04 18:44:35 +02:00
|
|
|
};
|
2006-07-10 13:44:43 +02:00
|
|
|
|
|
|
|
|
2006-07-11 18:28:15 +02:00
|
|
|
/* Compares only the schema part of the identifier */
|
|
|
|
bool
|
|
|
|
event_basic_db_equal(LEX_STRING db, Event_basic *et);
|
|
|
|
|
|
|
|
/* Compares the whole identifier*/
|
|
|
|
bool
|
|
|
|
event_basic_identifier_equal(LEX_STRING db, LEX_STRING name, Event_basic *b);
|
|
|
|
|
|
|
|
|
2006-06-27 08:48:50 +02:00
|
|
|
#endif /* _EVENT_DATA_OBJECTS_H_ */
|