Skip to content
Projects
Groups
Snippets
Help
Loading...
Help
Support
Keyboard shortcuts
?
Submit feedback
Contribute to GitLab
Sign in / Register
Toggle navigation
M
mariadb
Project overview
Project overview
Details
Activity
Releases
Repository
Repository
Files
Commits
Branches
Tags
Contributors
Graph
Compare
Issues
0
Issues
0
List
Boards
Labels
Milestones
Merge Requests
0
Merge Requests
0
Analytics
Analytics
Repository
Value Stream
Wiki
Wiki
Snippets
Snippets
Members
Members
Collapse sidebar
Close sidebar
Activity
Graph
Create a new issue
Commits
Issue Boards
Open sidebar
Kirill Smelkov
mariadb
Commits
d1673c15
Commit
d1673c15
authored
Dec 01, 2006
by
kostja@bodhi.local
Browse files
Options
Browse Files
Download
Plain Diff
Merge bk-internal.mysql.com:/home/bk/mysql-5.1
into bodhi.local:/opt/local/work/mysql-5.1-runtime
parents
6dc7e55e
21f3ce8f
Changes
27
Show whitespace changes
Inline
Side-by-side
Showing
27 changed files
with
1969 additions
and
964 deletions
+1969
-964
mysql-test/r/read_only.result
mysql-test/r/read_only.result
+51
-1
mysql-test/r/read_only_innodb.result
mysql-test/r/read_only_innodb.result
+18
-0
mysql-test/r/rpl_read_only.result
mysql-test/r/rpl_read_only.result
+113
-0
mysql-test/t/read_only.test
mysql-test/t/read_only.test
+108
-2
mysql-test/t/read_only_innodb.test
mysql-test/t/read_only_innodb.test
+43
-0
mysql-test/t/rpl_read_only-slave.opt
mysql-test/t/rpl_read_only-slave.opt
+1
-0
mysql-test/t/rpl_read_only.test
mysql-test/t/rpl_read_only.test
+105
-0
server-tools/instance-manager/commands.cc
server-tools/instance-manager/commands.cc
+171
-119
server-tools/instance-manager/commands.h
server-tools/instance-manager/commands.h
+65
-39
server-tools/instance-manager/guardian.cc
server-tools/instance-manager/guardian.cc
+259
-284
server-tools/instance-manager/guardian.h
server-tools/instance-manager/guardian.h
+51
-74
server-tools/instance-manager/instance.cc
server-tools/instance-manager/instance.cc
+417
-213
server-tools/instance-manager/instance.h
server-tools/instance-manager/instance.h
+134
-37
server-tools/instance-manager/instance_map.cc
server-tools/instance-manager/instance_map.cc
+120
-97
server-tools/instance-manager/instance_map.h
server-tools/instance-manager/instance_map.h
+14
-47
server-tools/instance-manager/instance_options.h
server-tools/instance-manager/instance_options.h
+0
-1
server-tools/instance-manager/manager.cc
server-tools/instance-manager/manager.cc
+120
-34
server-tools/instance-manager/manager.h
server-tools/instance-manager/manager.h
+7
-1
server-tools/instance-manager/user_map.cc
server-tools/instance-manager/user_map.cc
+10
-10
sql/handler.cc
sql/handler.cc
+13
-0
sql/lock.cc
sql/lock.cc
+17
-0
sql/set_var.cc
sql/set_var.cc
+65
-1
sql/set_var.h
sql/set_var.h
+14
-0
sql/sql_class.cc
sql/sql_class.cc
+14
-0
sql/sql_class.h
sql/sql_class.h
+10
-2
sql/sql_prepare.cc
sql/sql_prepare.cc
+1
-2
tests/mysql_client_test.c
tests/mysql_client_test.c
+28
-0
No files found.
mysql-test/r/read_only.result
View file @
d1673c15
...
...
@@ -39,6 +39,56 @@ delete t1 from t1,t3 where t1.a=t3.a;
drop table t1;
insert into t1 values(1);
ERROR HY000: The MySQL server is running with the --read-only option so it cannot execute this statement
set global read_only=0;
lock table t1 write;
lock table t2 write;
set global read_only=1;
ERROR HY000: Can't execute the given command because you have active locked tables or an active transaction
unlock tables ;
set global read_only=1;
select @@global.read_only;
@@global.read_only
0
unlock tables ;
select @@global.read_only;
@@global.read_only
1
set global read_only=0;
lock table t1 read;
lock table t2 read;
set global read_only=1;
ERROR HY000: Can't execute the given command because you have active locked tables or an active transaction
unlock tables ;
set global read_only=1;
select @@global.read_only;
@@global.read_only
0
unlock tables ;
select @@global.read_only;
@@global.read_only
1
set global read_only=0;
BEGIN;
BEGIN;
set global read_only=1;
ERROR HY000: Can't execute the given command because you have active locked tables or an active transaction
ROLLBACK;
set global read_only=1;
select @@global.read_only;
@@global.read_only
1
ROLLBACK;
set global read_only=0;
flush tables with read lock;
set global read_only=1;
unlock tables;
set global read_only=0;
flush tables with read lock;
set global read_only=1;
select @@global.read_only;
@@global.read_only
1
unlock tables;
set global read_only=0;
drop table t1,t2;
drop user test@localhost;
set global read_only=0;
mysql-test/r/read_only_innodb.result
0 → 100644
View file @
d1673c15
DROP TABLE IF EXISTS table_11733 ;
grant CREATE, SELECT, DROP on *.* to test@localhost;
set global read_only=0;
create table table_11733 (a int) engine=InnoDb;
BEGIN;
insert into table_11733 values(11733);
set global read_only=1;
select @@global.read_only;
@@global.read_only
1
select * from table_11733 ;
a
11733
COMMIT;
ERROR HY000: The MySQL server is running with the --read-only option so it cannot execute this statement
set global read_only=0;
drop table table_11733 ;
drop user test@localhost;
mysql-test/r/rpl_read_only.result
0 → 100644
View file @
d1673c15
stop slave;
drop table if exists t1,t2,t3,t4,t5,t6,t7,t8,t9;
reset master;
reset slave;
drop table if exists t1,t2,t3,t4,t5,t6,t7,t8,t9;
start slave;
create table t1(a int) engine=InnoDB;
create table t2(a int) engine=MyISAM;
insert into t1 values(1001);
insert into t2 values(2001);
set global read_only=1;
select @@read_only;
@@read_only
1
select * from t1;
a
1001
select * from t2;
a
2001
select @@read_only;
@@read_only
0
select * from t1;
a
1001
select * from t2;
a
2001
set global read_only=0;
BEGIN;
insert into t1 values(1002);
insert into t2 values(2002);
BEGIN;
insert into t1 values(1003);
insert into t2 values(2003);
set global read_only=1;
COMMIT;
COMMIT;
ERROR HY000: The MySQL server is running with the --read-only option so it cannot execute this statement
set global read_only=0;
insert into t1 values(1004);
insert into t2 values(2004);
select * from t1;
a
1001
1002
1004
select * from t2;
a
2001
2002
2003
2004
select * from t1;
a
1001
1002
1004
select * from t2;
a
2001
2002
2003
2004
set global read_only=1;
select @@read_only;
@@read_only
1
show create table t1;
Table Create Table
t1 CREATE TABLE `t1` (
`a` int(11) DEFAULT NULL
) ENGINE=InnoDB DEFAULT CHARSET=latin1
show create table t2;
Table Create Table
t2 CREATE TABLE `t2` (
`a` int(11) DEFAULT NULL
) ENGINE=MyISAM DEFAULT CHARSET=latin1
insert into t1 values(1005);
insert into t2 values(2005);
select * from t1;
a
1001
1002
1004
1005
select * from t2;
a
2001
2002
2003
2004
2005
select * from t1;
a
1001
1002
1004
1005
select * from t2;
a
2001
2002
2003
2004
2005
insert into t1 values(1006);
ERROR HY000: The MySQL server is running with the --read-only option so it cannot execute this statement
insert into t2 values(2006);
ERROR HY000: The MySQL server is running with the --read-only option so it cannot execute this statement
drop table t1;
drop table t2;
mysql-test/t/read_only.test
View file @
d1673c15
...
...
@@ -101,8 +101,114 @@ drop table t1;
--
error
1290
insert
into
t1
values
(
1
);
#
# BUG#11733: COMMITs should not happen if read-only is set
#
# LOCK TABLE ... WRITE / READ_ONLY
# - is an error in the same connection
# - is ok in a different connection
connection
default
;
drop
table
t1
,
t2
;
drop
user
test
@
localhost
;
set
global
read_only
=
0
;
lock
table
t1
write
;
connection
con1
;
lock
table
t2
write
;
connection
default
;
--
error
ER_LOCK_OR_ACTIVE_TRANSACTION
set
global
read_only
=
1
;
unlock
tables
;
# The following call blocks until con1 releases the write lock.
# Blocking is expected.
send
set
global
read_only
=
1
;
connection
con1
;
--
sleep
1
select
@@
global
.
read_only
;
unlock
tables
;
--
sleep
1
select
@@
global
.
read_only
;
connection
default
;
reap
;
# LOCK TABLE ... READ / READ_ONLY
# - is an error in the same connection
# - is ok in a different connection
connection
default
;
set
global
read_only
=
0
;
lock
table
t1
read
;
connection
con1
;
lock
table
t2
read
;
connection
default
;
--
error
ER_LOCK_OR_ACTIVE_TRANSACTION
set
global
read_only
=
1
;
unlock
tables
;
# The following call blocks until con1 releases the read lock.
# Blocking is a limitation, and could be improved.
send
set
global
read_only
=
1
;
connection
con1
;
--
sleep
1
select
@@
global
.
read_only
;
unlock
tables
;
--
sleep
1
select
@@
global
.
read_only
;
connection
default
;
reap
;
# pending transaction / READ_ONLY
# - is an error in the same connection
# - is ok in a different connection
connection
default
;
set
global
read_only
=
0
;
BEGIN
;
connection
con1
;
BEGIN
;
connection
default
;
--
error
ER_LOCK_OR_ACTIVE_TRANSACTION
set
global
read_only
=
1
;
ROLLBACK
;
set
global
read_only
=
1
;
connection
con1
;
select
@@
global
.
read_only
;
ROLLBACK
;
# Verify that FLUSH TABLES WITH READ LOCK do not block READ_ONLY
# - in the same SUPER connection
# - in another SUPER connection
connection
default
;
set
global
read_only
=
0
;
flush
tables
with
read
lock
;
set
global
read_only
=
1
;
unlock
tables
;
connect
(
root2
,
localhost
,
root
,,
test
);
connection
default
;
set
global
read_only
=
0
;
flush
tables
with
read
lock
;
connection
root2
;
set
global
read_only
=
1
;
connection
default
;
select
@@
global
.
read_only
;
unlock
tables
;
# Cleanup
connection
default
;
set
global
read_only
=
0
;
drop
table
t1
,
t2
;
drop
user
test
@
localhost
;
mysql-test/t/read_only_innodb.test
0 → 100644
View file @
d1673c15
# should work with embedded server after mysqltest is fixed
--
source
include
/
not_embedded
.
inc
--
source
include
/
have_innodb
.
inc
#
# BUG#11733: COMMITs should not happen if read-only is set
#
--
disable_warnings
DROP
TABLE
IF
EXISTS
table_11733
;
--
enable_warnings
# READ_ONLY does nothing to SUPER users
# so we use a non-SUPER one:
grant
CREATE
,
SELECT
,
DROP
on
*.*
to
test
@
localhost
;
connect
(
con1
,
localhost
,
test
,,
test
);
connection
default
;
set
global
read_only
=
0
;
# Any transactional engine will do
create
table
table_11733
(
a
int
)
engine
=
InnoDb
;
connection
con1
;
BEGIN
;
insert
into
table_11733
values
(
11733
);
connection
default
;
set
global
read_only
=
1
;
connection
con1
;
select
@@
global
.
read_only
;
select
*
from
table_11733
;
--
error
ER_OPTION_PREVENTS_STATEMENT
COMMIT
;
connection
default
;
set
global
read_only
=
0
;
drop
table
table_11733
;
drop
user
test
@
localhost
;
mysql-test/t/rpl_read_only-slave.opt
0 → 100644
View file @
d1673c15
--innodb
mysql-test/t/rpl_read_only.test
0 → 100644
View file @
d1673c15
# Test case for BUG #11733
--
source
include
/
master
-
slave
.
inc
--
source
include
/
have_innodb
.
inc
# Setting the master readonly :
# - the variable @@readonly is not replicated on the slave
connect
(
master2
,
127.0
.
0.1
,
test
,,
test
,
$MASTER_MYPORT
,);
connect
(
slave2
,
127.0
.
0.1
,
test
,,
test
,
$SLAVE_MYPORT
,);
connection
master1
;
create
table
t1
(
a
int
)
engine
=
InnoDB
;
create
table
t2
(
a
int
)
engine
=
MyISAM
;
insert
into
t1
values
(
1001
);
insert
into
t2
values
(
2001
);
connection
master
;
set
global
read_only
=
1
;
connection
master1
;
select
@@
read_only
;
select
*
from
t1
;
select
*
from
t2
;
sync_slave_with_master
;
select
@@
read_only
;
select
*
from
t1
;
select
*
from
t2
;
# - replication of transactions
connection
master
;
set
global
read_only
=
0
;
connection
master1
;
BEGIN
;
insert
into
t1
values
(
1002
);
insert
into
t2
values
(
2002
);
connection
master2
;
BEGIN
;
insert
into
t1
values
(
1003
);
insert
into
t2
values
(
2003
);
connection
master
;
set
global
read_only
=
1
;
connection
master1
;
## works even with read_only=1, because master1 is root
COMMIT
;
connection
master2
;
--
error
ER_OPTION_PREVENTS_STATEMENT
COMMIT
;
connection
master
;
set
global
read_only
=
0
;
connection
master1
;
insert
into
t1
values
(
1004
);
insert
into
t2
values
(
2004
);
select
*
from
t1
;
select
*
from
t2
;
sync_slave_with_master
;
select
*
from
t1
;
select
*
from
t2
;
# Setting the slave readonly : replication will pass
#
connection
slave1
;
set
global
read_only
=
1
;
connection
slave
;
select
@@
read_only
;
# Make sure the replicated table is also transactional
show
create
table
t1
;
# Make sure the replicated table is not transactional
show
create
table
t2
;
connection
master
;
insert
into
t1
values
(
1005
);
insert
into
t2
values
(
2005
);
select
*
from
t1
;
select
*
from
t2
;
sync_slave_with_master
;
connection
slave
;
select
*
from
t1
;
select
*
from
t2
;
# Non root user can not write on the slave
connection
slave2
;
--
error
ER_OPTION_PREVENTS_STATEMENT
insert
into
t1
values
(
1006
);
--
error
ER_OPTION_PREVENTS_STATEMENT
insert
into
t2
values
(
2006
);
## Cleanup
connection
master
;
drop
table
t1
;
drop
table
t2
;
sync_slave_with_master
;
server-tools/instance-manager/commands.cc
View file @
d1673c15
...
...
@@ -29,6 +29,7 @@
#include "guardian.h"
#include "instance_map.h"
#include "log.h"
#include "manager.h"
#include "messages.h"
#include "mysqld_error.h"
#include "mysql_manager_error.h"
...
...
@@ -36,8 +37,11 @@
#include "priv.h"
#include "protocol.h"
/**************************************************************************
{{{ Static functions.
**************************************************************************/
/*
/*
*
modify_defaults_to_im_error -- a map of error codes of
mysys::modify_defaults_file() into Instance Manager error codes.
*/
...
...
@@ -46,38 +50,25 @@ static const int modify_defaults_to_im_error[]= { 0, ER_OUT_OF_RESOURCES,
ER_ACCESS_OPTION_FILE
};
/*
Add a string to a buffer
.
/*
*
Parse version number from the version string
.
SYNOPSIS
p
ut_to_buff
()
buff buffer to add the string
str string to add
position offset in the buff to add a string
p
arse_version_number
()
version_str
version
version_size
DESCRIPTION
TODO
Function to add a string to the buffer. It is different from
store_to_protocol_packet, which is used in the protocol.cc.
The last one also stores the length of the string in a special way.
This is required for MySQL client/server protocol support only.
TODO: Move this function to Instance_options and parse version number
only once.
RETURN
0 - ok
1 - error occured
NOTE: This function is used only in SHOW INSTANCE STATUS statement at the
moment.
*/
static
inline
int
put_to_buff
(
Buffer
*
buff
,
const
char
*
str
,
uint
*
position
)
{
uint
len
=
strlen
(
str
);
if
(
buff
->
append
(
*
position
,
str
,
len
))
return
1
;
*
position
+=
len
;
return
0
;
}
static
int
parse_version_number
(
const
char
*
version_str
,
char
*
version
,
uint
version_size
)
{
...
...
@@ -102,6 +93,9 @@ static int parse_version_number(const char *version_str, char *version,
return
0
;
}
/**************************************************************************
}}}
**************************************************************************/
/**************************************************************************
Implementation of Instance_name.
...
...
@@ -122,7 +116,7 @@ Instance_name::Instance_name(const LEX_STRING *name)
Implementation of Show_instances.
**************************************************************************/
/*
/*
*
Implementation of SHOW INSTANCES statement.
Possible error codes:
...
...
@@ -172,7 +166,6 @@ int Show_instances::write_data(st_net *net)
Instance
*
instance
;
Instance_map
::
Iterator
iterator
(
instance_map
);
instance_map
->
guardian
->
lock
();
instance_map
->
lock
();
while
((
instance
=
iterator
.
next
()))
...
...
@@ -180,20 +173,25 @@ int Show_instances::write_data(st_net *net)
Buffer
send_buf
;
/* buffer for packets */
uint
pos
=
0
;
instance
->
lock
();
const
char
*
instance_name
=
instance
->
options
.
instance_name
.
str
;
const
char
*
state_name
=
instance
_map
->
get_instance_state_name
(
instance
);
const
char
*
state_name
=
instance
->
get_state_name
(
);
if
(
store_to_protocol_packet
(
&
send_buf
,
instance_name
,
&
pos
)
||
store_to_protocol_packet
(
&
send_buf
,
state_name
,
&
pos
)
||
my_net_write
(
net
,
send_buf
.
buffer
,
pos
))
{
err_status
=
TRUE
;
break
;
}
instance
->
unlock
();
if
(
err_status
)
break
;
}
instance_map
->
unlock
();
instance_map
->
guardian
->
unlock
();
return
err_status
?
ER_OUT_OF_RESOURCES
:
0
;
}
...
...
@@ -203,7 +201,7 @@ int Show_instances::write_data(st_net *net)
Implementation of Flush_instances.
**************************************************************************/
/*
/*
*
Implementation of FLUSH INSTANCES statement.
Possible error codes:
...
...
@@ -213,36 +211,19 @@ int Show_instances::write_data(st_net *net)
int
Flush_instances
::
execute
(
st_net
*
net
,
ulong
connection_id
)
{
instance_map
->
guardian
->
lock
();
instance_map
->
lock
();
if
(
instance_map
->
is_there_active_instance
())
{
instance_map
->
unlock
();
instance_map
->
guardian
->
unlock
();
return
ER_THERE_IS_ACTIVE_INSTACE
;
}
if
(
instance_map
->
flush_instances
())
{
instance_map
->
unlock
();
instance_map
->
guardian
->
unlock
();
if
(
Manager
::
flush_instances
())
return
ER_OUT_OF_RESOURCES
;
}
instance_map
->
unlock
();
instance_map
->
guardian
->
unlock
();
return
net_send_ok
(
net
,
connection_id
,
NULL
)
?
ER_OUT_OF_RESOURCES
:
0
;
}
/**************************************************************************
Implementation of
Abstract_i
nstance_cmd.
Implementation of
I
nstance_cmd.
**************************************************************************/
Abstract_instance_cmd
::
Abstract_instance_cmd
(
const
LEX_STRING
*
instance_name_arg
)
:
instance_name
(
instance_name_arg
)
Instance_cmd
::
Instance_cmd
(
const
LEX_STRING
*
instance_name_arg
)
:
instance_name
(
instance_name_arg
)
{
/*
MT-NOTE: we can not make a search for Instance object here,
...
...
@@ -251,14 +232,25 @@ Abstract_instance_cmd::Abstract_instance_cmd(const LEX_STRING *instance_name_arg
}
/**************************************************************************
Implementation of Abstract_instance_cmd.
**************************************************************************/
Abstract_instance_cmd
::
Abstract_instance_cmd
(
const
LEX_STRING
*
instance_name_arg
)
:
Instance_cmd
(
instance_name_arg
)
{
}
int
Abstract_instance_cmd
::
execute
(
st_net
*
net
,
ulong
connection_id
)
{
int
err_code
;
Instance
*
instance
;
instance_map
->
lock
();
{
Instance
*
instance
=
instance_map
->
find
(
get_instance_name
());
instance
=
instance_map
->
find
(
get_instance_name
());
if
(
!
instance
)
{
...
...
@@ -266,10 +258,12 @@ int Abstract_instance_cmd::execute(st_net *net, ulong connection_id)
return
ER_BAD_INSTANCE_NAME
;
}
instance
->
lock
();
instance_map
->
unlock
();
err_code
=
execute_impl
(
net
,
instance
);
}
instance
_map
->
unlock
();
instance
->
unlock
();
if
(
!
err_code
)
err_code
=
send_ok_response
(
net
,
connection_id
);
...
...
@@ -288,7 +282,7 @@ Show_instance_status::Show_instance_status(const LEX_STRING *instance_name_arg)
}
/*
/*
*
Implementation of SHOW INSTANCE STATUS statement.
Possible error codes:
...
...
@@ -363,19 +357,14 @@ int Show_instance_status::write_data(st_net *net, Instance *instance)
char
version_num_buf
[
MAX_VERSION_LENGTH
];
uint
pos
=
0
;
const
char
*
state_name
;
const
char
*
state_name
=
instance
->
get_state_name
()
;
const
char
*
version_tag
=
"unknown"
;
const
char
*
version_num
=
"unknown"
;
const
char
*
mysqld_compatible_status
;
instance_map
->
guardian
->
lock
();
state_name
=
instance_map
->
get_instance_state_name
(
instance
);
mysqld_compatible_status
=
instance
->
is_mysqld_compatible
()
?
"yes"
:
"no"
;
instance_map
->
guardian
->
unlock
();
const
char
*
mysqld_compatible_status
=
instance
->
is_mysqld_compatible
()
?
"yes"
:
"no"
;
if
(
instance
->
options
.
mysqld_version
)
{
if
(
parse_version_number
(
instance
->
options
.
mysqld_version
,
version_num_buf
,
sizeof
(
version_num_buf
)))
return
ER_OUT_OF_RESOURCES
;
...
...
@@ -409,7 +398,7 @@ Show_instance_options::Show_instance_options(
}
/*
/*
*
Implementation of SHOW INSTANCE OPTIONS statement.
Possible error codes:
...
...
@@ -505,23 +494,33 @@ Start_instance::Start_instance(const LEX_STRING *instance_name_arg)
}
/*
/*
*
Implementation of START INSTANCE statement.
Possible error codes:
ER_BAD_INSTANCE_NAME The instance with the given name does not exist
ER_OUT_OF_RESOURCES Not enough resources to complete the operation
ER_INSTANCE_MISCONFIGURED The instance configuration is invalid
ER_INSTANCE_ALREADY_STARTED The instance is already started
ER_CANNOT_START_INSTANCE The instance could not have been started
TODO: as soon as this method operates only with Instance, we probably
should introduce a new method (execute_stop_instance()) in Instance and
just call it from here.
*/
int
Start_instance
::
execute_impl
(
st_net
*
/* net */
,
Instance
*
instance
)
{
int
err_code
;
if
(
!
instance
->
is_configured
())
return
ER_INSTANCE_MISCONFIGURED
;
if
((
err_code
=
instance
->
start
()))
return
err_code
;
if
(
instance
->
is_active
())
return
ER_INSTANCE_ALREADY_STARTED
;
if
(
instance
->
start_mysqld
())
return
ER_CANNOT_START_INSTANCE
;
i
f
(
!
(
instance
->
options
.
nonguarded
))
instance_map
->
guardian
->
guard
(
instance
);
i
nstance
->
reset_stat
();
instance
->
set_state
(
Instance
::
NOT_STARTED
);
return
0
;
}
...
...
@@ -546,25 +545,26 @@ Stop_instance::Stop_instance(const LEX_STRING *instance_name_arg)
}
/*
/*
*
Implementation of STOP INSTANCE statement.
Possible error codes:
ER_BAD_INSTANCE_NAME The instance with the given name does not exist
ER_OUT_OF_RESOURCES Not enough resources to complete the operation
TODO: as soon as this method operates only with Instance, we probably
should introduce a new method (execute_stop_instance()) in Instance and
just call it from here.
*/
int
Stop_instance
::
execute_impl
(
st_net
*
/* net */
,
Instance
*
instance
)
{
int
err_code
;
if
(
!
instance
->
is_active
())
return
ER_INSTANCE_IS_NOT_STARTED
;
if
(
!
(
instance
->
options
.
nonguarded
))
instance_map
->
guardian
->
stop_guard
(
instance
);
if
((
err_code
=
instance
->
stop
()))
return
err_code
;
instance
->
set_state
(
Instance
::
STOPPED
);
return
0
;
return
instance
->
stop_mysqld
()
?
ER_STOP_INSTANCE
:
0
;
}
...
...
@@ -582,12 +582,12 @@ int Stop_instance::send_ok_response(st_net *net, ulong connection_id)
**************************************************************************/
Create_instance
::
Create_instance
(
const
LEX_STRING
*
instance_name_arg
)
:
instance_name
(
instance_name_arg
)
:
Instance_cmd
(
instance_name_arg
)
{
}
/*
/*
*
This operation initializes Create_instance object.
SYNOPSIS
...
...
@@ -604,7 +604,7 @@ bool Create_instance::init(const char **text)
}
/*
/*
*
This operation parses CREATE INSTANCE options.
SYNOPSIS
...
...
@@ -724,7 +724,7 @@ bool Create_instance::parse_args(const char **text)
}
/*
/*
*
Implementation of CREATE INSTANCE statement.
Possible error codes:
...
...
@@ -736,6 +736,7 @@ bool Create_instance::parse_args(const char **text)
int
Create_instance
::
execute
(
st_net
*
net
,
ulong
connection_id
)
{
int
err_code
;
Instance
*
instance
;
/* Check that the name is valid and there is no instance with such name. */
...
...
@@ -761,17 +762,26 @@ int Create_instance::execute(st_net *net, ulong connection_id)
return
err_code
;
}
instance
=
instance_map
->
find
(
get_instance_name
());
DBUG_ASSERT
(
instance
);
if
((
err_code
=
create_instance_in_file
(
get_instance_name
(),
&
options
)))
{
Instance
*
instance
=
instance_map
->
find
(
get_instance_name
());
if
(
instance
)
instance_map
->
remove_instance
(
instance
);
/* instance is deleted here. */
instance_map
->
unlock
();
return
err_code
;
}
/*
CREATE INSTANCE must not lead to start instance, even if it guarded.
TODO: The problem however is that if Instance Manager restarts after
creating instance, the instance will be restarted (see also BUG#19718).
*/
instance
->
set_state
(
Instance
::
STOPPED
);
/* That's all. */
instance_map
->
unlock
();
...
...
@@ -790,12 +800,12 @@ int Create_instance::execute(st_net *net, ulong connection_id)
**************************************************************************/
Drop_instance
::
Drop_instance
(
const
LEX_STRING
*
instance_name_arg
)
:
Abstract_i
nstance_cmd
(
instance_name_arg
)
:
I
nstance_cmd
(
instance_name_arg
)
{
}
/*
/*
*
Implementation of DROP INSTANCE statement.
Possible error codes:
...
...
@@ -804,14 +814,38 @@ Drop_instance::Drop_instance(const LEX_STRING *instance_name_arg)
ER_OUT_OF_RESOURCES Not enough resources to complete the operation
*/
int
Drop_instance
::
execute
_impl
(
st_net
*
/* net */
,
Instance
*
instance
)
int
Drop_instance
::
execute
(
st_net
*
net
,
ulong
connection_id
)
{
int
err_code
;
Instance
*
instance
;
/* Lock Guardian, then Instance_map. */
instance_map
->
lock
();
/* Find an instance. */
instance
=
instance_map
->
find
(
get_instance_name
());
if
(
!
instance
)
{
instance_map
->
unlock
();
return
ER_BAD_INSTANCE_NAME
;
}
instance
->
lock
();
/* Check that the instance is offline. */
if
(
instance_map
->
guardian
->
is_active
(
instance
))
if
(
instance
->
is_active
())
{
instance
->
unlock
();
instance_map
->
unlock
();
return
ER_DROP_ACTIVE_INSTANCE
;
}
/* Try to remove instance from the file. */
err_code
=
modify_defaults_file
(
Options
::
Main
::
config_file
,
NULL
,
NULL
,
get_instance_name
()
->
str
,
MY_REMOVE_SECTION
);
...
...
@@ -824,27 +858,30 @@ int Drop_instance::execute_impl(st_net * /* net */, Instance *instance)
(
const
char
*
)
get_instance_name
()
->
str
,
(
const
char
*
)
Options
::
Main
::
config_file
,
(
int
)
err_code
);
}
if
(
err_code
)
instance
->
unlock
();
instance_map
->
unlock
();
return
modify_defaults_to_im_error
[
err_code
];
}
/*
Remove instance from the instance map hash and Guardian's list
. */
/*
Unlock the instance before destroy
. */
if
(
!
instance
->
options
.
nonguarded
)
instance_map
->
guardian
->
stop_guard
(
instance
);
instance
->
unlock
();
if
((
err_code
=
instance
->
stop
()))
return
err_code
;
/*
Remove instance from the instance map
(the instance will be also destroyed here).
*/
instance_map
->
remove_instance
(
instance
);
return
0
;
}
/* Unlock the instance map. */
instance_map
->
unlock
();
/* That's all: send ok. */
int
Drop_instance
::
send_ok_response
(
st_net
*
net
,
ulong
connection_id
)
{
if
(
net_send_ok
(
net
,
connection_id
,
"Instance dropped"
))
return
ER_OUT_OF_RESOURCES
;
...
...
@@ -867,7 +904,7 @@ Show_instance_log::Show_instance_log(const LEX_STRING *instance_name_arg,
}
/*
/*
*
Implementation of SHOW INSTANCE LOG statement.
Possible error codes:
...
...
@@ -1012,7 +1049,7 @@ Show_instance_log_files::Show_instance_log_files
}
/*
/*
*
Implementation of SHOW INSTANCE LOG FILES statement.
Possible error codes:
...
...
@@ -1133,7 +1170,7 @@ int Show_instance_log_files::write_data(st_net *net, Instance *instance)
Implementation of Abstract_option_cmd.
**************************************************************************/
/*
/*
*
Instance_options_list -- a data class representing a list of options for
some instance.
*/
...
...
@@ -1251,7 +1288,7 @@ bool Abstract_option_cmd::init(const char **text)
}
/*
/*
*
Correct the option file. The "skip" option is used to remove the found
option.
...
...
@@ -1290,8 +1327,8 @@ int Abstract_option_cmd::correct_file(Instance *instance, Named_value *option,
}
/*
Implementation of SET statement
.
/*
*
Lock Instance Map and call execute_impl()
.
Possible error codes:
ER_BAD_INSTANCE_NAME The instance with the given name does not exist
...
...
@@ -1341,6 +1378,11 @@ Abstract_option_cmd::get_instance_options_list(const LEX_STRING *instance_name)
}
/**
Skeleton implementation of option-management command.
MT-NOTE: Instance Map is locked before calling this operation.
*/
int
Abstract_option_cmd
::
execute_impl
(
st_net
*
net
,
ulong
connection_id
)
{
int
err_code
=
0
;
...
...
@@ -1352,12 +1394,18 @@ int Abstract_option_cmd::execute_impl(st_net *net, ulong connection_id)
Instance_options_list
*
lst
=
(
Instance_options_list
*
)
hash_element
(
&
instance_options_map
,
i
);
bool
instance_is_active
;
lst
->
instance
=
instance_map
->
find
(
lst
->
get_instance_name
());
if
(
!
lst
->
instance
)
return
ER_BAD_INSTANCE_NAME
;
if
(
instance_map
->
guardian
->
is_active
(
lst
->
instance
))
lst
->
instance
->
lock
();
instance_is_active
=
lst
->
instance
->
is_active
();
lst
->
instance
->
unlock
();
if
(
instance_is_active
)
return
ER_INSTANCE_IS_ACTIVE
;
}
...
...
@@ -1368,6 +1416,8 @@ int Abstract_option_cmd::execute_impl(st_net *net, ulong connection_id)
Instance_options_list
*
lst
=
(
Instance_options_list
*
)
hash_element
(
&
instance_options_map
,
i
);
lst
->
instance
->
lock
();
for
(
int
j
=
0
;
j
<
lst
->
options
.
get_size
();
++
j
)
{
Named_value
option
=
lst
->
options
.
get_element
(
j
);
...
...
@@ -1377,6 +1427,8 @@ int Abstract_option_cmd::execute_impl(st_net *net, ulong connection_id)
break
;
}
lst
->
instance
->
unlock
();
if
(
err_code
)
break
;
}
...
...
@@ -1392,7 +1444,7 @@ int Abstract_option_cmd::execute_impl(st_net *net, ulong connection_id)
Implementation of Set_option.
**************************************************************************/
/*
/*
*
This operation parses SET options.
SYNOPSIS
...
...
@@ -1566,7 +1618,7 @@ int Set_option::process_option(Instance *instance, Named_value *option)
Implementation of Unset_option.
**************************************************************************/
/*
/*
*
This operation parses UNSET options.
SYNOPSIS
...
...
@@ -1662,7 +1714,7 @@ bool Unset_option::parse_args(const char **text)
}
/*
/*
*
Implementation of UNSET statement.
Possible error codes:
...
...
server-tools/instance-manager/commands.h
View file @
d1673c15
...
...
@@ -30,7 +30,7 @@
#endif
/*
/*
*
Print all instances of this instance manager.
Grammar: SHOW INSTANCES
*/
...
...
@@ -50,7 +50,7 @@ private:
};
/*
/*
*
Reread configuration file and refresh internal cache.
Grammar: FLUSH INSTANCES
*/
...
...
@@ -66,11 +66,50 @@ public:
};
/*
/**
Base class for Instance-specific commands
(commands that operate on one instance).
Instance_cmd extends Command class by:
- an attribute for storing instance name;
- code to initialize instance name in constructor;
- an accessor to get instance name.
*/
class
Instance_cmd
:
public
Command
{
public:
Instance_cmd
(
const
LEX_STRING
*
instance_name_arg
);
protected:
inline
const
LEX_STRING
*
get_instance_name
()
const
{
return
instance_name
.
get_str
();
}
private:
Instance_name
instance_name
;
};
/**
Abstract class for Instance-specific commands.
Abstract_instance_cmd extends Instance_cmd by providing a common
framework for writing command-implementations. Basically, the class
implements Command::execute() pure virtual function in the following
way:
- Lock Instance_map;
- Get an instance by name. Return an error, if there is no such
instance;
- Lock the instance;
- Unlock Instance_map;
- Call execute_impl(), which should be implemented in derived class;
- Unlock the instance;
- Send response to the client and return error status.
*/
class
Abstract_instance_cmd
:
public
Comman
d
class
Abstract_instance_cmd
:
public
Instance_cm
d
{
public:
Abstract_instance_cmd
(
const
LEX_STRING
*
instance_name_arg
);
...
...
@@ -79,29 +118,24 @@ public:
virtual
int
execute
(
st_net
*
net
,
ulong
connection_id
);
protected:
/* MT-NOTE: this operation is called under acquired Instance_map's lock. */
/**
This operation is intended to contain command-specific implementation.
MT-NOTE: this operation is called under acquired Instance's lock.
*/
virtual
int
execute_impl
(
st_net
*
net
,
Instance
*
instance
)
=
0
;
/*
/*
*
This operation is invoked on successful return of execute_impl() and is
intended to send closing data.
MT-NOTE: this operation is called under released Instance
_map
's lock.
MT-NOTE: this operation is called under released Instance's lock.
*/
virtual
int
send_ok_response
(
st_net
*
net
,
ulong
connection_id
)
=
0
;
protected:
inline
const
LEX_STRING
*
get_instance_name
()
const
{
return
instance_name
.
get_str
();
}
private:
Instance_name
instance_name
;
};
/*
/*
*
Print status of an instance.
Grammar: SHOW INSTANCE STATUS <instance_name>
*/
...
...
@@ -121,7 +155,7 @@ private:
};
/*
/*
*
Print options of chosen instance.
Grammar: SHOW INSTANCE OPTIONS <instance_name>
*/
...
...
@@ -141,7 +175,7 @@ private:
};
/*
/*
*
Start an instance.
Grammar: START INSTANCE <instance_name>
*/
...
...
@@ -157,7 +191,7 @@ protected:
};
/*
/*
*
Stop an instance.
Grammar: STOP INSTANCE <instance_name>
*/
...
...
@@ -173,12 +207,12 @@ protected:
};
/*
/*
*
Create an instance.
Grammar: CREATE INSTANCE <instance_name> [<options>]
*/
class
Create_instance
:
public
Comman
d
class
Create_instance
:
public
Instance_cm
d
{
public:
Create_instance
(
const
LEX_STRING
*
instance_name_arg
);
...
...
@@ -189,22 +223,15 @@ public:
protected:
virtual
int
execute
(
st_net
*
net
,
ulong
connection_id
);
inline
const
LEX_STRING
*
get_instance_name
()
const
{
return
instance_name
.
get_str
();
}
private:
bool
parse_args
(
const
char
**
text
);
private:
Instance_name
instance_name
;
Named_value_arr
options
;
};
/*
/*
*
Drop an instance.
Grammar: DROP INSTANCE <instance_name>
...
...
@@ -213,18 +240,17 @@ private:
is removed from the instance map.
*/
class
Drop_instance
:
public
Abstract_i
nstance_cmd
class
Drop_instance
:
public
I
nstance_cmd
{
public:
Drop_instance
(
const
LEX_STRING
*
instance_name_arg
);
protected:
virtual
int
execute_impl
(
st_net
*
net
,
Instance
*
instance
);
virtual
int
send_ok_response
(
st_net
*
net
,
ulong
connection_id
);
virtual
int
execute
(
st_net
*
net
,
ulong
connection_id
);
};
/*
/*
*
Print requested part of the log.
Grammar:
SHOW <instance_name> LOG {ERROR | SLOW | GENERAL} size[, offset_from_end]
...
...
@@ -252,7 +278,7 @@ private:
};
/*
/*
*
Shows the list of the log files, used by an instance.
Grammar: SHOW <instance_name> LOG FILES
*/
...
...
@@ -272,7 +298,7 @@ private:
};
/*
/*
*
Abstract class for option-management commands.
*/
...
...
@@ -312,7 +338,7 @@ private:
};
/*
/*
*
Set an option for the instance.
Grammar: SET instance_name.option[=option_value][, ...]
*/
...
...
@@ -329,7 +355,7 @@ protected:
};
/*
/*
*
Remove option of the instance.
Grammar: UNSET instance_name.option[, ...]
*/
...
...
@@ -346,7 +372,7 @@ protected:
};
/*
/*
*
Syntax error command.
This command is issued if parser reported a syntax error. We need it to
...
...
server-tools/instance-manager/guardian.cc
View file @
d1673c15
...
...
@@ -28,101 +28,126 @@
#include "instance_map.h"
#include "log.h"
#include "mysql_manager_error.h"
#include "options.h"
const
char
*
Guardian
::
get_instance_state_name
(
enum_instance_state
state
)
{
switch
(
state
)
{
case
NOT_STARTED
:
return
"offline"
;
case
STARTING
:
return
"starting"
;
case
STARTED
:
return
"online"
;
case
JUST_CRASHED
:
return
"failed"
;
/*************************************************************************
{{{ Constructor & destructor.
*************************************************************************/
case
CRASHED
:
return
"crashed"
;
case
CRASHED_AND_ABANDONED
:
return
"abandoned"
;
case
STOPPING
:
return
"stopping"
;
}
/**
Guardian constructor.
return
NULL
;
/* just to ignore compiler warning. */
}
SYNOPSIS
Guardian()
thread_registry_arg
instance_map_arg
/* {{{ Constructor & destructor. */
DESCRIPTION
Nominal contructor intended for assigning references and initialize
trivial objects. Real initialization is made by init() method.
*/
Guardian
::
Guardian
(
Thread_registry
*
thread_registry_arg
,
Instance_map
*
instance_map_arg
,
uint
monitoring_interval_arg
)
:
stopped
(
FALSE
),
monitoring_interval
(
monitoring_interval_arg
),
Instance_map
*
instance_map_arg
)
:
shutdown_requested
(
FALSE
),
stopped
(
FALSE
),
thread_registry
(
thread_registry_arg
),
instance_map
(
instance_map_arg
),
shutdown_requested
(
FALSE
)
instance_map
(
instance_map_arg
)
{
pthread_mutex_init
(
&
LOCK_guardian
,
0
);
pthread_cond_init
(
&
COND_guardian
,
0
);
init_alloc_root
(
&
alloc
,
MEM_ROOT_BLOCK_SIZE
,
0
);
}
Guardian
::~
Guardian
()
{
/* delay guardian destruction to the moment when no one needs it */
pthread_mutex_lock
(
&
LOCK_guardian
);
free_root
(
&
alloc
,
MYF
(
0
));
pthread_mutex_unlock
(
&
LOCK_guardian
);
/*
NOTE: it's necessary to synchronize here, because Guiardian thread can be
still alive an hold the mutex (because it is detached and we have no
control over it).
*/
lock
();
unlock
();
pthread_mutex_destroy
(
&
LOCK_guardian
);
pthread_cond_destroy
(
&
COND_guardian
);
}
/* }}} */
/*************************************************************************
}}}
*************************************************************************/
/**
Send request to stop Guardian.
SYNOPSIS
request_shutdown()
*/
void
Guardian
::
request_shutdown
()
{
pthread_mutex_lock
(
&
LOCK_guardian
);
/* STOP Instances or just clean up Guardian repository */
stop_instances
();
lock
();
shutdown_requested
=
TRUE
;
pthread_mutex_unlock
(
&
LOCK_guardian
);
unlock
();
ping
();
}
void
Guardian
::
process_instance
(
Instance
*
instance
,
GUARD_NODE
*
current_node
,
LIST
**
guarded_instances
,
LIST
*
node
)
/**
Process an instance.
SYNOPSIS
process_instance()
instance a pointer to the instance for processing
MT-NOTE:
- the given instance must be locked before calling this operation;
- Guardian must be locked before calling this operation.
*/
void
Guardian
::
process_instance
(
Instance
*
instance
)
{
uint
waitchild
=
(
uint
)
Instance
::
DEFAULT_SHUTDOWN_DELAY
;
/* The amount of times, Guardian attempts to restart an instance */
int
restart_retry
=
100
;
time_t
current_time
=
time
(
NULL
);
if
(
current_node
->
state
==
STOPPING
)
if
(
instance
->
get_state
()
==
Instance
::
STOPPING
)
{
waitchild
=
instance
->
options
.
get_shutdown_delay
();
/* This brach is executed during shutdown. */
/*
this returns TRUE if and only if an instance was stopped for sure
*/
/*
This returns TRUE if and only if an instance was stopped for sure.
*/
if
(
instance
->
is_crashed
())
*
guarded_instances
=
list_delete
(
*
guarded_instances
,
node
);
else
if
(
(
uint
)
(
current_time
-
current_node
->
last_checked
)
>
waitchild
)
{
log_info
(
"Guardian: '%s' stopped."
,
(
const
char
*
)
instance
->
get_name
()
->
str
);
instance
->
set_state
(
Instance
::
STOPPED
);
}
else
if
((
uint
)
(
current_time
-
instance
->
last_checked
)
>=
instance
->
options
.
get_shutdown_delay
())
{
log_info
(
"Guardian: '%s' hasn't stopped within %d secs."
,
(
const
char
*
)
instance
->
get_name
()
->
str
,
(
int
)
instance
->
options
.
get_shutdown_delay
());
instance
->
kill_mysqld
(
SIGKILL
);
/*
Later we do node= node->next. This is ok, as we are only removing
the node from the list. The pointer to the next one is still valid.
*/
*
guarded_instances
=
list_delete
(
*
guarded_instances
,
node
);
log_info
(
"Guardian: pretend that '%s' is killed."
,
(
const
char
*
)
instance
->
get_name
()
->
str
);
instance
->
set_state
(
Instance
::
STOPPED
);
}
else
{
log_info
(
"Guardian: waiting for '%s' to stop (%d secs left)."
,
(
const
char
*
)
instance
->
get_name
()
->
str
,
(
int
)
(
instance
->
options
.
get_shutdown_delay
()
-
current_time
+
instance
->
last_checked
));
}
return
;
...
...
@@ -133,68 +158,72 @@ void Guardian::process_instance(Instance *instance,
/* The instance can be contacted on it's port */
/* If STARTING also check that pidfile has been created */
if
(
current_node
->
state
==
STARTING
&&
current_node
->
instance
->
options
.
load_pid
()
==
0
)
if
(
instance
->
get_state
()
==
Instance
::
STARTING
&&
instance
->
options
.
load_pid
()
==
0
)
{
/* Pid file not created yet, don't go to STARTED state yet */
}
else
if
(
current_node
->
state
!=
STARTED
)
else
if
(
instance
->
get_state
()
!=
Instance
::
STARTED
)
{
/* clear status fields */
log_info
(
"Guardian: '%s' is running, set state to STARTED."
,
(
const
char
*
)
instance
->
options
.
instance_name
.
str
);
current_node
->
restart_counter
=
0
;
current_node
->
crash_moment
=
0
;
current_node
->
state
=
STARTED
;
instance
->
reset_stat
();
instance
->
set_state
(
Instance
::
STARTED
);
}
}
else
{
switch
(
current_node
->
state
)
{
case
NOT_STARTED
:
switch
(
instance
->
get_state
()
)
{
case
Instance
:
:
NOT_STARTED
:
log_info
(
"Guardian: starting '%s'..."
,
(
const
char
*
)
instance
->
options
.
instance_name
.
str
);
/* NOTE, set state to STARTING _before_ start() is called */
current_node
->
state
=
STARTING
;
instance
->
start
();
current_node
->
last_checked
=
current_time
;
break
;
case
STARTED
:
/* fallthrough */
case
STARTING
:
/* let the instance start or crash */
if
(
instance
->
is_crashed
())
{
current_node
->
crash_moment
=
current_time
;
current_node
->
last_checked
=
current_time
;
current_node
->
state
=
JUST_CRASHED
;
/* NOTE: set state to STARTING _before_ start() is called. */
instance
->
set_state
(
Instance
::
STARTING
);
instance
->
last_checked
=
current_time
;
instance
->
start_mysqld
();
return
;
case
Instance
:
:
STARTED
:
/* fallthrough */
case
Instance
:
:
STARTING
:
/* let the instance start or crash */
if
(
!
instance
->
is_crashed
())
return
;
instance
->
crash_moment
=
current_time
;
instance
->
last_checked
=
current_time
;
instance
->
set_state
(
Instance
::
JUST_CRASHED
);
/* fallthrough -- restart an instance immediately */
}
else
break
;
case
JUST_CRASHED
:
if
(
current_time
-
current_node
->
crash_moment
<=
2
)
case
Instance
:
:
JUST_CRASHED
:
if
(
current_time
-
instance
->
crash_moment
<=
2
)
{
if
(
instance
->
is_crashed
())
{
instance
->
start
();
instance
->
start
_mysqld
();
log_info
(
"Guardian: starting '%s'..."
,
(
const
char
*
)
instance
->
options
.
instance_name
.
str
);
}
}
else
current_node
->
state
=
CRASHED
;
break
;
case
CRASHED
:
/* just regular restarts */
if
(
current_time
-
current_node
->
last_checked
>
monitoring_interval
)
{
if
((
current_node
->
restart_counter
<
restart_retry
))
instance
->
set_state
(
Instance
::
CRASHED
);
return
;
case
Instance
:
:
CRASHED
:
/* just regular restarts */
if
(
current_time
-
instance
->
last_checked
<=
Options
::
Main
::
monitoring_interval
)
return
;
if
(
instance
->
restart_counter
<
restart_retry
)
{
if
(
instance
->
is_crashed
())
{
instance
->
start
();
current_nod
e
->
last_checked
=
current_time
;
current_node
->
restart_counter
++
;
instance
->
start_mysqld
();
instanc
e
->
last_checked
=
current_time
;
log_info
(
"Guardian: restarting '%s'..."
,
(
const
char
*
)
instance
->
options
.
instance_name
.
str
);
}
...
...
@@ -204,12 +233,15 @@ void Guardian::process_instance(Instance *instance,
log_info
(
"Guardian: can not start '%s'. "
"Abandoning attempts to (re)start it"
,
(
const
char
*
)
instance
->
options
.
instance_name
.
str
);
current_node
->
state
=
CRASHED_AND_ABANDONED
;
}
instance
->
set_state
(
Instance
::
CRASHED_AND_ABANDONED
);
}
break
;
case
CRASHED_AND_ABANDONED
:
break
;
/* do nothing */
return
;
case
Instance
:
:
CRASHED_AND_ABANDONED
:
return
;
/* do nothing */
default:
DBUG_ASSERT
(
0
);
}
...
...
@@ -217,56 +249,78 @@ void Guardian::process_instance(Instance *instance,
}
/*
/*
*
Main function of Guardian thread.
SYNOPSIS
run()
DESCRIPTION
Check for all guarded instances and restart them if needed. If everything
is fine go and sleep for some time.
Check for all guarded instances and restart them if needed.
*/
void
Guardian
::
run
()
{
Instance
*
instance
;
LIST
*
node
;
struct
timespec
timeout
;
log_info
(
"Guardian: started."
);
thread_registry
->
register_thread
(
&
thread_info
);
pthread_mutex_lock
(
&
LOCK_guardian
);
/* Loop, until all instances were shut down at the end. */
while
(
true
)
{
Instance_map
::
Iterator
instances_it
(
instance_map
);
Instance
*
instance
;
bool
all_instances_stopped
=
TRUE
;
instance_map
->
lock
();
/* loop, until all instances were shut down at the end */
while
(
!
(
shutdown_requested
&&
(
guarded_instances
==
NULL
)))
while
((
instance
=
instances_it
.
next
()))
{
node
=
guarded_instances
;
instance
->
lock
()
;
while
(
node
!=
NULL
)
if
(
!
instance
->
is_guarded
()
||
instance
->
get_state
()
==
Instance
::
STOPPED
)
{
GUARD_NODE
*
current_node
=
(
GUARD_NODE
*
)
node
->
data
;
instance
=
((
GUARD_NODE
*
)
node
->
data
)
->
instance
;
process_instance
(
instance
,
current_node
,
&
guarded_instances
,
node
);
instance
->
unlock
();
continue
;
}
process_instance
(
instance
);
node
=
node
->
next
;
if
(
instance
->
get_state
()
!=
Instance
::
STOPPED
)
all_instances_stopped
=
FALSE
;
instance
->
unlock
();
}
instance_map
->
unlock
();
lock
();
if
(
shutdown_requested
&&
all_instances_stopped
)
{
log_info
(
"Guardian: all guarded mysqlds stopped."
);
stopped
=
TRUE
;
unlock
();
break
;
}
timeout
.
tv_sec
=
time
(
NULL
)
+
monitoring_interval
;
timeout
.
tv_sec
=
time
(
NULL
)
+
Options
::
Main
::
monitoring_interval
;
timeout
.
tv_nsec
=
0
;
/* check the loop predicate before sleeping */
if
(
!
(
shutdown_requested
&&
(
!
(
guarded_instances
))))
thread_registry
->
cond_timedwait
(
&
thread_info
,
&
COND_guardian
,
&
LOCK_guardian
,
&
timeout
);
unlock
();
}
log_info
(
"Guardian: stopped."
);
stopped
=
TRUE
;
pthread_mutex_unlock
(
&
LOCK_guardian
);
/* now, when the Guardian is stopped we can stop the IM */
/* Now, when the Guardian is stopped we can stop the IM. */
thread_registry
->
unregister_thread
(
&
thread_info
);
thread_registry
->
request_shutdown
();
...
...
@@ -274,129 +328,65 @@ void Guardian::run()
}
int
Guardian
::
is_stopped
()
/**
Return the value of stopped flag.
*/
bool
Guardian
::
is_stopped
()
{
int
var
;
pthread_mutex_lock
(
&
LOCK_guardian
);
lock
();
var
=
stopped
;
pthread_mutex_unlock
(
&
LOCK_guardian
);
unlock
();
return
var
;
}
/*
Initialize the list of guarded instances: loop through the Instance_map and
add all of the instances, which don't have 'nonguarded' option specified.
SYNOPSIS
Guardian::init()
NOTE: The operation should be invoked with the following locks acquired:
- Guardian;
- Instance_map;
/**
Wake up Guardian thread.
RETURN
0 - ok
1 - error occurred
MT-NOTE: though usually the mutex associated with condition variable should
be acquired before signalling the variable, here this is not needed.
Signalling under locked mutex is used to avoid lost signals. In the current
logic however locking mutex does not guarantee that the signal will not be
lost.
*/
int
Guardian
::
init
()
void
Guardian
::
ping
()
{
Instance
*
instance
;
Instance_map
::
Iterator
iterator
(
instance_map
);
/* clear the list of guarded instances */
free_root
(
&
alloc
,
MYF
(
0
));
init_alloc_root
(
&
alloc
,
MEM_ROOT_BLOCK_SIZE
,
0
);
guarded_instances
=
NULL
;
while
((
instance
=
iterator
.
next
()))
{
if
(
instance
->
options
.
nonguarded
)
continue
;
if
(
guard
(
instance
,
TRUE
))
/* do not lock guardian */
return
1
;
}
return
0
;
pthread_cond_signal
(
&
COND_guardian
);
}
/*
Add instance to the Guardian list
/*
*
Prepare list of instances.
SYNOPSIS
guard()
instance the instance to be guarded
nolock whether we prefer do not lock Guardian here,
but use external locking instead
DESCRIPTION
The instance is added to the guarded instances list. Usually guard() is
called after we start an instance.
init()
RETURN
0 - ok
1 - error occurred
MT-NOTE: Instance Map must be locked before calling the operation.
*/
int
Guardian
::
guard
(
Instance
*
instance
,
bool
nolock
)
void
Guardian
::
init
(
)
{
LIST
*
node
;
GUARD_NODE
*
content
;
node
=
(
LIST
*
)
alloc_root
(
&
alloc
,
sizeof
(
LIST
));
content
=
(
GUARD_NODE
*
)
alloc_root
(
&
alloc
,
sizeof
(
GUARD_NODE
));
if
((
!
(
node
))
||
(
!
(
content
)))
return
1
;
/* we store the pointers to instances from the instance_map's MEM_ROOT */
content
->
instance
=
instance
;
content
->
restart_counter
=
0
;
content
->
crash_moment
=
0
;
content
->
state
=
NOT_STARTED
;
node
->
data
=
(
void
*
)
content
;
if
(
nolock
)
guarded_instances
=
list_add
(
guarded_instances
,
node
);
else
{
pthread_mutex_lock
(
&
LOCK_guardian
);
guarded_instances
=
list_add
(
guarded_instances
,
node
);
pthread_mutex_unlock
(
&
LOCK_guardian
);
}
return
0
;
}
/*
TODO: perhaps it would make sense to create a pool of the LIST nodeents
and give them upon request. Now we are loosing a bit of memory when
guarded instance was stopped and then restarted (since we cannot free just
a piece of the MEM_ROOT).
*/
int
Guardian
::
stop_guard
(
Instance
*
instance
)
{
LIST
*
node
;
pthread_mutex_lock
(
&
LOCK_guardian
);
node
=
find_instance_node
(
instance
);
Instance
*
instance
;
Instance_map
::
Iterator
iterator
(
instance_map
);
if
(
node
!=
NULL
)
guarded_instances
=
list_delete
(
guarded_instances
,
node
);
while
((
instance
=
iterator
.
next
()))
{
instance
->
lock
();
pthread_mutex_unlock
(
&
LOCK_guardian
);
instance
->
reset_stat
();
instance
->
set_state
(
Instance
::
NOT_STARTED
);
/* if there is nothing to delete it is also fine */
return
0
;
instance
->
unlock
();
}
}
/*
/**
An internal method which is called at shutdown to unregister instances and
attempt to stop them if requested.
...
...
@@ -409,86 +399,71 @@ int Guardian::stop_guard(Instance *instance)
accordingly.
NOTE
Guardian object should be locked by the call
ing function
.
Guardian object should be locked by the call
er
.
RETURN
0 - ok
1 - error occurred
*/
int
Guardian
::
stop_instances
()
void
Guardian
::
stop_instances
()
{
LIST
*
node
;
node
=
guarded_instances
;
while
(
node
!=
NULL
)
Instance_map
::
Iterator
instances_it
(
instance_map
);
Instance
*
instance
;
instance_map
->
lock
();
while
((
instance
=
instances_it
.
next
()))
{
instance
->
lock
();
if
(
!
instance
->
is_guarded
()
||
instance
->
get_state
()
==
Instance
::
STOPPED
)
{
GUARD_NODE
*
current_node
=
(
GUARD_NODE
*
)
node
->
data
;
instance
->
unlock
();
continue
;
}
/*
If instance is running or was running (and now probably hanging),
request stop.
*/
if
(
current_node
->
instance
->
is_mysqld_running
()
||
(
current_node
->
state
==
STARTED
))
if
(
instance
->
is_mysqld_running
()
||
instance
->
get_state
()
==
Instance
::
STARTED
)
{
current_node
->
state
=
STOPPING
;
current_nod
e
->
last_checked
=
time
(
NULL
);
instance
->
set_state
(
Instance
::
STOPPING
)
;
instanc
e
->
last_checked
=
time
(
NULL
);
}
else
/* otherwise remove it from the list */
guarded_instances
=
list_delete
(
guarded_instances
,
node
);
/* But try to kill it anyway. Just in case */
current_node
->
instance
->
kill_mysqld
(
SIGTERM
);
node
=
node
->
next
;
{
/* Otherwise mark it as STOPPED. */
instance
->
set_state
(
Instance
::
STOPPED
);
}
return
0
;
}
void
Guardian
::
lock
()
{
pthread_mutex_lock
(
&
LOCK_guardian
);
}
void
Guardian
::
unlock
()
{
pthread_mutex_unlock
(
&
LOCK_guardian
);
}
LIST
*
Guardian
::
find_instance_node
(
Instance
*
instance
)
{
LIST
*
node
=
guarded_instances
;
/* Request mysqld to stop. */
while
(
node
!=
NULL
)
{
/*
We compare only pointers, as we always use pointers from the
instance_map's MEM_ROOT.
*/
if
(((
GUARD_NODE
*
)
node
->
data
)
->
instance
==
instance
)
return
node
;
instance
->
kill_mysqld
(
SIGTERM
);
node
=
node
->
next
;
instance
->
unlock
()
;
}
return
NULL
;
instance_map
->
unlock
()
;
}
bool
Guardian
::
is_active
(
Instance
*
instance
)
{
bool
guarded
;
lock
();
/**
Lock Guardian.
*/
guarded
=
find_instance_node
(
instance
)
!=
NULL
;
void
Guardian
::
lock
()
{
pthread_mutex_lock
(
&
LOCK_guardian
);
}
/* is_running() can take a long time, so let's unlock mutex first. */
unlock
();
if
(
guarded
)
return
true
;
/**
Unlock Guardian.
*/
return
instance
->
is_mysqld_running
();
void
Guardian
::
unlock
()
{
pthread_mutex_unlock
(
&
LOCK_guardian
);
}
server-tools/instance-manager/guardian.h
View file @
d1673c15
...
...
@@ -17,10 +17,12 @@
Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA */
#include
"thread_registry.h"
#include
<my_global.h>
#include <my_sys.h>
#include <my_list.h>
#include "thread_registry.h"
#if defined(__GNUC__) && defined(USE_PRAGMA_INTERFACE)
#pragma interface
#endif
...
...
@@ -28,7 +30,6 @@
class
Instance
;
class
Instance_map
;
class
Thread_registry
;
struct
GUARD_NODE
;
/**
The guardian thread is responsible for monitoring and restarting of guarded
...
...
@@ -38,97 +39,73 @@ struct GUARD_NODE;
class
Guardian
:
public
Thread
{
public:
/* states of an instance */
enum
enum_instance_state
{
NOT_STARTED
=
1
,
STARTING
,
STARTED
,
JUST_CRASHED
,
CRASHED
,
CRASHED_AND_ABANDONED
,
STOPPING
};
/*
The Guardian list node structure. Guardian utilizes it to store
guarded instances plus some additional info.
*/
Guardian
(
Thread_registry
*
thread_registry_arg
,
Instance_map
*
instance_map_arg
);
~
Guardian
();
struct
GUARD_NODE
{
Instance
*
instance
;
/* state of an instance (i.e. STARTED, CRASHED, etc.) */
enum_instance_state
state
;
/* the amount of attemts to restart instance (cleaned up at success) */
int
restart_counter
;
/* triggered at a crash */
time_t
crash_moment
;
/* General time field. Used to provide timeouts (at shutdown and restart) */
time_t
last_checked
;
};
/* Return client state name. */
static
const
char
*
get_instance_state_name
(
enum_instance_state
state
);
void
init
();
Guardian
(
Thread_registry
*
thread_registry_arg
,
Instance_map
*
instance_map_arg
,
uint
monitoring_interval_arg
);
virtual
~
Guardian
();
/* Initialize or refresh the list of guarded instances */
int
init
();
/* Request guardian shutdown. Stop instances if needed */
public:
void
request_shutdown
();
/* Start instance protection */
int
guard
(
Instance
*
instance
,
bool
nolock
=
FALSE
);
/* Stop instance protection */
int
stop_guard
(
Instance
*
instance
);
/* Returns TRUE if guardian thread is stopped */
int
is_stopped
();
bool
is_stopped
();
void
lock
();
void
unlock
();
/*
Return an internal list node for the given instance if the instance is
managed by Guardian. Otherwise, return NULL.
void
ping
();
MT-NOTE: must be called under acquired lock.
*/
LIST
*
find_instance_node
(
Instance
*
instance
);
protected:
virtual
void
run
();
private:
void
stop_instances
();
/* The operation is used to check if the instance is active or not. */
bool
is_active
(
Instance
*
instance
);
void
process_instance
(
Instance
*
instance
);
private:
/*
Return state of the given instance list node. The pointer must specify
a valid list node.
LOCK_guardian protectes the members in this section:
- shutdown_requested;
- stopped;
Also, it is used for COND_guardian.
*/
inline
enum_instance_state
get_instance_state
(
LIST
*
instance_node
);
protected:
/* Main funtion of the thread */
virtual
void
run
();
pthread_mutex_t
LOCK_guardian
;
public:
/*
Guardian's main loop waits on this condition. So, it should be signalled
each time, when instance state has been changed and we want Guardian to
wake up.
TODO: Change this to having data-scoped conditions, i.e. conditions,
which indicate that some data has been changed.
*/
pthread_cond_t
COND_guardian
;
private:
/* Prepares Guardian shutdown. Stops instances is needed */
int
stop_instances
();
/* check instance state and act accordingly */
void
process_instance
(
Instance
*
instance
,
GUARD_NODE
*
current_node
,
LIST
**
guarded_instances
,
LIST
*
elem
);
/*
This variable is set to TRUE, when Manager thread is shutting down.
The flag is used by Guardian thread to understand that it's time to
finish.
*/
bool
shutdown_requested
;
/*
This flag is set to TRUE on shutdown by Guardian thread, when all guarded
mysqlds are stopped.
int
stopped
;
The flag is used in the Manager thread to wait for Guardian to stop all
mysqlds.
*/
bool
stopped
;
private:
pthread_mutex_t
LOCK_guardian
;
Thread_info
thread_info
;
int
monitoring_interval
;
Thread_registry
*
thread_registry
;
Instance_map
*
instance_map
;
LIST
*
guarded_instances
;
MEM_ROOT
alloc
;
/* this variable is set to TRUE when we want to stop Guardian thread */
bool
shutdown_requested
;
};
inline
Guardian
::
enum_instance_state
Guardian
::
get_instance_state
(
LIST
*
instance_node
)
{
return
((
GUARD_NODE
*
)
instance_node
->
data
)
->
state
;
}
private:
Guardian
(
const
Guardian
&
);
Guardian
&
operator
=
(
const
Guardian
&
);
};
#endif
/* INCLUDES_MYSQL_INSTANCE_MANAGER_GUARDIAN_H */
server-tools/instance-manager/instance.cc
View file @
d1673c15
...
...
@@ -36,7 +36,9 @@
#include "thread_registry.h"
#include "instance_map.h"
/* {{{ Platform-specific functions. */
/*************************************************************************
{{{ Platform-specific functions.
*************************************************************************/
#ifndef __WIN__
typedef
pid_t
My_process_info
;
...
...
@@ -44,34 +46,6 @@ typedef pid_t My_process_info;
typedef
PROCESS_INFORMATION
My_process_info
;
#endif
/*
Proxy thread is a simple way to avoid all pitfalls of the threads
implementation in the OS (e.g. LinuxThreads). With such a thread we
don't have to process SIGCHLD, which is a tricky business if we want
to do it in a portable way.
*/
class
Instance_monitor
:
public
Thread
{
public:
Instance_monitor
(
Instance
*
instance_arg
)
:
instance
(
instance_arg
)
{}
protected:
virtual
void
run
();
void
start_and_monitor_instance
(
Instance_options
*
old_instance_options
,
Instance_map
*
instance_map
,
Thread_registry
*
thread_registry
);
private:
Instance
*
instance
;
};
void
Instance_monitor
::
run
()
{
start_and_monitor_instance
(
&
instance
->
options
,
Manager
::
get_instance_map
(),
Manager
::
get_thread_registry
());
delete
this
;
}
/*
Wait for an instance
...
...
@@ -285,113 +259,149 @@ int kill(pid_t pid, int signum)
}
#endif
/* }}} */
/*************************************************************************
}}}
*************************************************************************/
/* {{{ Static constants. */
/*************************************************************************
{{{ Static constants.
*************************************************************************/
const
LEX_STRING
Instance
::
DFLT_INSTANCE_NAME
=
{
C_STRING_WITH_LEN
(
"mysqld"
)
};
/* }}} */
/*************************************************************************
}}}
*************************************************************************/
/*
Fork child, exec an instance and monitor it.
/*************************************************************************
{{{ Instance Monitor thread.
*************************************************************************/
SYNOPSIS
start_and_monitor_instance()
old_instance_options Pointer to the options of the instance to be
launched. This info is likely to become obsolete
when function returns from wait_process()
instance_map Pointer to the instance_map. We use it to protect
the instance from deletion, while we are working
with it.
/**
Proxy thread is a simple way to avoid all pitfalls of the threads
implementation in the OS (e.g. LinuxThreads). With such a thread we
don't have to process SIGCHLD, which is a tricky business if we want
to do it in a portable way.
DESCRIPTION
Fork a child, then exec and monitor it. When the child is dead,
find appropriate instance (for this purpose we save its name),
set appropriate flags and wake all threads waiting for instance
to stop.
NOTE
A separate thread for starting/monitoring instance is a simple way
to avoid all pitfalls of the threads implementation in the OS (e.g.
LinuxThreads). For one, with such a thread we don't have to process
SIGCHLD, which is a tricky business if we want to do it in a
portable way.
Instance Monitor Thread forks a child process, execs mysqld and waits for
the child to die.
RETURN
Function returns no value
Instance Monitor assumes that the monitoring instance will not be dropped.
This is guaranteed by having flag monitoring_thread_active and
Instance::is_active() operation.
*/
void
Instance_monitor
::
start_and_monitor_instance
(
Instance_options
*
old_instance_options
,
Instance_map
*
instance_map
,
Thread_registry
*
thread_registry
)
class
Instance_monitor
:
public
Thread
{
Instance_name
instance_name
(
&
old_instance_options
->
instance_name
);
Instance
*
current_instance
;
My_process_info
process_info
;
Thread_info
thread_info
;
public:
Instance_monitor
(
Instance
*
instance_arg
)
:
instance
(
instance_arg
)
{}
protected:
virtual
void
run
();
void
start_and_monitor_instance
();
private:
Instance
*
instance
;
};
void
Instance_monitor
::
run
()
{
start_and_monitor_instance
();
delete
this
;
}
void
Instance_monitor
::
start_and_monitor_instance
()
{
Thread_registry
*
thread_registry
=
Manager
::
get_thread_registry
();
Guardian
*
guardian
=
Manager
::
get_guardian
();
My_process_info
mysqld_process_info
;
Thread_info
monitor_thread_info
;
log_info
(
"Instance '%s': Monitor: started."
,
(
const
char
*
)
instance
->
get_name
()
->
str
);
if
(
!
old_instance_options
->
nonguarded
)
{
/*
Register thread in Thread_registry to wait for it to stop on shutdown
only if instance is guarded. If instance is guarded, the thread will not
finish, because nonguarded instances are not stopped on shutdown
.
For guarded instance register the thread in Thread_registry to wait for
the thread to stop on shutdown (nonguarded instances are not stopped on
shutdown, so the thread will no finish)
.
*/
thread_registry
->
register_thread
(
&
thread_info
,
FALSE
);
if
(
instance
->
is_guarded
())
{
thread_registry
->
register_thread
(
&
monitor_thread_info
,
FALSE
);
}
/*
Lock instance map to guarantee that no instances are deleted during
strmake() and execv() calls.
*/
instance_map
->
lock
();
/*
Save the instance name in the case if Instance object we
are using is destroyed. (E.g. by "FLUSH INSTANCES")
*/
/* Starting mysqld. */
log_info
(
"Instance '%s': Monitor: starting mysqld..."
,
(
const
char
*
)
instance
->
get_name
()
->
str
);
if
(
start_process
(
old_instance_options
,
&
process_info
))
if
(
start_process
(
&
instance
->
options
,
&
mysqld_
process_info
))
{
instance_map
->
unlock
();
return
;
/* error is logged */
instance
->
lock
();
instance
->
monitoring_thread_active
=
FALSE
;
instance
->
unlock
();
return
;
}
/* allow users to delete instances */
instance_map
->
unlock
();
/* Waiting for mysqld to die. */
log_info
(
"Instance '%s': Monitor: waiting for mysqld to stop..."
,
(
const
char
*
)
instance
->
get_name
()
->
str
);
wait_process
(
&
process_info
);
/* Don't check for return value. */
wait_process
(
&
mysqld_
process_info
);
/* Don't check for return value. */
instance_map
->
lock
();
log_info
(
"Instance '%s': Monitor: mysqld stopped."
,
(
const
char
*
)
instance
->
get_name
()
->
str
);
current_instance
=
instance_map
->
find
(
instance_name
.
get_str
());
/* Update instance status. */
if
(
current_instance
)
current_instance
->
set_crash_flag_n_wake_all
();
instance
->
lock
();
instance_map
->
unlock
();
if
(
instance
->
is_guarded
())
thread_registry
->
unregister_thread
(
&
monitor_thread_info
);
i
f
(
!
old_instance_options
->
nonguarded
)
thread_registry
->
unregister_thread
(
&
thread_info
)
;
i
nstance
->
crashed
=
TRUE
;
instance
->
monitoring_thread_active
=
FALSE
;
log_info
(
"Instance '%s': Monitor: finished."
,
(
const
char
*
)
instance
->
get_name
()
->
str
);
instance
->
unlock
();
/* Wake up guardian. */
guardian
->
ping
();
}
/**************************************************************************
}}}
**************************************************************************/
/**************************************************************************
{{{ Static operations.
**************************************************************************/
/**
The operation is intended to check whether string is a well-formed
instance name or not.
SYNOPSIS
is_name_valid()
name string to check
RETURN
TRUE string is a valid instance name
FALSE string is not a valid instance name
TODO: Move to Instance_name class: Instance_name::is_valid().
*/
bool
Instance
::
is_name_valid
(
const
LEX_STRING
*
name
)
{
...
...
@@ -405,21 +415,83 @@ bool Instance::is_name_valid(const LEX_STRING *name)
}
/**
The operation is intended to check if the given instance name is
mysqld-compatible or not.
SYNOPSIS
is_mysqld_compatible_name()
name name to check
RETURN
TRUE name is mysqld-compatible
FALSE otherwise
TODO: Move to Instance_name class: Instance_name::is_mysqld_compatible().
*/
bool
Instance
::
is_mysqld_compatible_name
(
const
LEX_STRING
*
name
)
{
return
strcmp
(
name
->
str
,
DFLT_INSTANCE_NAME
.
str
)
==
0
;
}
/**
Return client state name. Must not be used outside the class.
Use Instance::get_state_name() instead.
*/
const
char
*
Instance
::
get_instance_state_name
(
enum_instance_state
state
)
{
switch
(
state
)
{
case
STOPPED
:
return
"offline"
;
case
NOT_STARTED
:
return
"not started"
;
case
STARTING
:
return
"starting"
;
case
STARTED
:
return
"online"
;
case
JUST_CRASHED
:
return
"failed"
;
case
CRASHED
:
return
"crashed"
;
case
CRASHED_AND_ABANDONED
:
return
"abandoned"
;
case
STOPPING
:
return
"stopping"
;
}
return
NULL
;
/* just to ignore compiler warning. */
}
/**************************************************************************
}}}
**************************************************************************/
/* {{{ Constructor & destructor */
/**************************************************************************
{{{ Initialization & deinitialization.
**************************************************************************/
Instance
::
Instance
()
:
crashed
(
FALSE
),
configured
(
FALSE
)
:
monitoring_thread_active
(
FALSE
),
crashed
(
FALSE
),
configured
(
FALSE
),
/* mysqld_compatible is initialized in init() */
state
(
NOT_STARTED
),
restart_counter
(
0
),
crash_moment
(
0
),
last_checked
(
0
)
{
pthread_mutex_init
(
&
LOCK_instance
,
0
);
pthread_cond_init
(
&
COND_instance_stopped
,
0
);
}
...
...
@@ -427,13 +499,11 @@ Instance::~Instance()
{
log_info
(
"Instance '%s': destroying..."
,
(
const
char
*
)
get_name
()
->
str
);
pthread_cond_destroy
(
&
COND_instance_stopped
);
pthread_mutex_destroy
(
&
LOCK_instance
);
}
/* }}} */
/*
/*
*
Initialize instance options.
SYNOPSIS
...
...
@@ -453,7 +523,7 @@ bool Instance::init(const LEX_STRING *name_arg)
}
/*
/*
*
Complete instance options initialization.
SYNOPSIS
...
...
@@ -474,7 +544,47 @@ bool Instance::complete_initialization()
*/
}
/*
/**************************************************************************
}}}
**************************************************************************/
/**************************************************************************
{{{ Instance: public interface implementation.
**************************************************************************/
/**
Determine if there is some activity with the instance.
SYNOPSIS
is_active()
DESCRIPTION
An instance is active if one of the following conditions is true:
- Instance-monitoring thread is running;
- Instance is guarded and its state is other than STOPPED;
- Corresponding mysqld-server accepts connections.
MT-NOTE: instance must be locked before calling the operation.
RETURN
TRUE - instance is active
FALSE - otherwise.
*/
bool
Instance
::
is_active
()
{
if
(
monitoring_thread_active
)
return
TRUE
;
if
(
is_guarded
()
&&
get_state
()
!=
STOPPED
)
return
TRUE
;
return
is_mysqld_running
();
}
/**
Determine if mysqld is accepting connections.
SYNOPSIS
...
...
@@ -484,7 +594,7 @@ bool Instance::complete_initialization()
Try to connect to mysqld with fake login/password to check whether it is
accepting connections or not.
MT-NOTE:
this operation must be called under acquired LOCK_instance
.
MT-NOTE:
instance must be locked before calling the operation
.
RETURN
TRUE - mysqld is alive and accept connections
...
...
@@ -508,8 +618,6 @@ bool Instance::is_mysqld_running()
if
(
!
port
&&
!
options
.
mysqld_socket
)
port
=
SERVER_DEFAULT_PORT
;
pthread_mutex_lock
(
&
LOCK_instance
);
mysql_init
(
&
mysql
);
/* try to connect to a server with a fake username/password pair */
if
(
mysql_real_connect
(
&
mysql
,
LOCAL_HOST
,
username
,
...
...
@@ -523,7 +631,6 @@ bool Instance::is_mysqld_running()
*/
log_error
(
"Instance '%s': was able to log into mysqld."
,
(
const
char
*
)
get_name
()
->
str
);
pthread_mutex_unlock
(
&
LOCK_instance
);
return_val
=
TRUE
;
/* server is alive */
}
else
...
...
@@ -531,145 +638,145 @@ bool Instance::is_mysqld_running()
sizeof
(
access_denied_message
)
-
1
));
mysql_close
(
&
mysql
);
pthread_mutex_unlock
(
&
LOCK_instance
);
return
return_val
;
}
/*
The method starts an instance.
/**
Start mysqld.
SYNOPSIS
start()
start_mysqld()
DESCRIPTION
Reset flags and start Instance Monitor thread, which will start mysqld.
MT-NOTE: instance must be locked before calling the operation.
RETURN
0 ok
ER_CANNOT_START_INSTANCE Cannot start instance
ER_INSTANCE_ALREADY_STARTED The instance on the specified port/socket
is already started
FALSE - ok
TRUE - could not start instance
*/
int
Instance
::
start
()
bool
Instance
::
start_mysqld
()
{
/* clear crash flag */
pthread_mutex_lock
(
&
LOCK_instance
);
crashed
=
FALSE
;
pthread_mutex_unlock
(
&
LOCK_instance
);
Instance_monitor
*
instance_monitor
;
/*
Prepare instance to start Instance Monitor thread.
NOTE: It's important to set these actions here in order to avoid
race conditions -- these actions must be done under acquired lock on
Instance.
*/
crashed
=
FALSE
;
monitoring_thread_active
=
TRUE
;
if
(
configured
&&
!
is_mysqld_running
())
{
Instance_monitor
*
instance_monitor
;
remove_pid
();
/* Create and start the Instance Monitor thread. */
instance_monitor
=
new
Instance_monitor
(
this
);
if
(
instance_monitor
==
NULL
||
instance_monitor
->
start
(
Thread
::
DETACHED
))
{
delete
instance_monitor
;
log_error
(
"Instance::start(): failed to create the monitoring thread"
" to start an instance"
);
return
ER_CANNOT_START_INSTANCE
;
}
/* The monitoring thread will delete itself when it's finished. */
monitoring_thread_active
=
FALSE
;
return
0
;
}
/* The instance is started already or misconfigured. */
return
configured
?
ER_INSTANCE_ALREADY_STARTED
:
ER_INSTANCE_MISCONFIGURED
;
}
/*
The method sets the crash flag and wakes all waiters on
COND_instance_stopped and COND_guardian
log_error
(
"Instance '%s': can not create instance monitor thread."
,
(
const
char
*
)
get_name
()
->
str
);
SYNOPSIS
set_crash_flag_n_wake_all()
return
TRUE
;
}
DESCRIPTION
The method is called when an instance is crashed or terminated.
In the former case it might indicate that guardian probably should
restart it.
++
restart_counter
;
RETURN
Function returns no value
*/
/* The Instance Monitor thread will delete itself when it's finished. */
void
Instance
::
set_crash_flag_n_wake_all
()
{
/* set instance state to crashed */
pthread_mutex_lock
(
&
LOCK_instance
);
crashed
=
TRUE
;
pthread_mutex_unlock
(
&
LOCK_instance
);
/*
Wake connection threads waiting for an instance to stop. This
is needed if a user issued command to stop an instance via
mysql connection. This is not the case if Guardian stop the thread.
*/
pthread_cond_signal
(
&
COND_instance_stopped
);
/* wake guardian */
pthread_cond_signal
(
&
Manager
::
get_guardian
()
->
COND_guardian
);
return
FALSE
;
}
/*
Stop
an instance
.
/*
*
Stop
mysqld
.
SYNOPSIS
stop()
stop
_mysqld
()
RETURN:
0 ok
ER_INSTANCE_IS_NOT_STARTED Looks like the instance it is not started
ER_STOP_INSTANCE mysql_shutdown reported an error
DESCRIPTION
Try to stop mysqld gracefully. Otherwise kill it with SIGKILL.
MT-NOTE: instance must be locked before calling the operation.
RETURN
FALSE - ok
TRUE - could not stop the instance
*/
int
Instance
::
stop
()
bool
Instance
::
stop_mysqld
()
{
struct
timespec
timeout
;
uint
waitchild
=
(
uint
)
DEFAULT_SHUTDOWN_DELAY
;
if
(
is_mysqld_running
())
{
waitchild
=
options
.
get_shutdown_delay
();
log_info
(
"Instance '%s': stopping mysqld..."
,
(
const
char
*
)
get_name
()
->
str
);
kill_mysqld
(
SIGTERM
);
/* sleep on condition to wait for SIGCHLD */
timeout
.
tv_sec
=
time
(
NULL
)
+
waitchild
;
timeout
.
tv_nsec
=
0
;
if
(
pthread_mutex_lock
(
&
LOCK_instance
))
return
ER_STOP_INSTANCE
;
while
(
options
.
load_pid
()
!=
0
)
/* while server isn't stopped */
if
(
!
wait_for_stop
())
{
int
status
;
status
=
pthread_cond_timedwait
(
&
COND_instance_stopped
,
&
LOCK_instance
,
&
timeout
);
if
(
status
==
ETIMEDOUT
||
status
==
ETIME
)
break
;
log_info
(
"Instance '%s': mysqld stopped gracefully."
,
(
const
char
*
)
get_name
()
->
str
);
return
FALSE
;
}
pthread_mutex_unlock
(
&
LOCK_instance
);
log_info
(
"Instance '%s': mysqld failed to stop gracefully within %d seconds."
,
(
const
char
*
)
get_name
()
->
str
,
(
int
)
options
.
get_shutdown_delay
());
log_info
(
"Instance'%s': killing mysqld..."
,
(
const
char
*
)
get_name
()
->
str
);
kill_mysqld
(
SIGKILL
);
return
0
;
if
(
!
wait_for_stop
())
{
log_info
(
"Instance '%s': mysqld has been killed."
,
(
const
char
*
)
get_name
()
->
str
);
return
FALSE
;
}
return
ER_INSTANCE_IS_NOT_STARTED
;
log_info
(
"Instance '%s': can not kill mysqld within %d seconds."
,
(
const
char
*
)
get_name
()
->
str
,
(
int
)
options
.
get_shutdown_delay
());
return
TRUE
;
}
/*
/*
*
Send signal to mysqld.
SYNOPSIS
kill_mysqld()
DESCRIPTION
Load pid from the pid file and send the given signal to that process.
If the signal is SIGKILL, remove the pid file after sending the signal.
MT-NOTE: instance must be locked before calling the operation.
TODO
This too low-level and OS-specific operation for public interface.
Also, it has some implicit behaviour for SIGKILL signal. Probably, we
should have the following public operations instead:
- start_mysqld() -- as is;
- stop_mysqld -- request mysqld to shutdown gracefully (send SIGTERM);
don't wait for complete shutdown;
- wait_for_stop() (or join_mysqld()) -- wait for mysqld to stop within
time interval;
- kill_mysqld() -- request to terminate mysqld; don't wait for
completion.
These operations should also be used in Guardian to manage instances.
*/
void
Instance
::
kill_mysqld
(
int
signum
)
...
...
@@ -707,27 +814,91 @@ void Instance::kill_mysqld(int signum)
}
}
/*
Return crashed flag.
SYNOPSIS
is_crashed()
RETURN
TRUE - mysqld crashed
FALSE - mysqld hasn't crashed yet
/**
Lock instance.
*/
bool
Instance
::
is_crashed
()
void
Instance
::
lock
()
{
bool
val
;
pthread_mutex_lock
(
&
LOCK_instance
);
val
=
crashed
;
}
/**
Unlock instance.
*/
void
Instance
::
unlock
()
{
pthread_mutex_unlock
(
&
LOCK_instance
);
return
val
;
}
/*
/**
Return instance state name.
SYNOPSIS
get_state_name()
DESCRIPTION
The operation returns user-friendly state name. The operation can be
used both for guarded and non-guarded instances.
MT-NOTE: instance must be locked before calling the operation.
TODO: Replace with the static get_state_name(state_code) function.
*/
const
char
*
Instance
::
get_state_name
()
{
if
(
!
is_configured
())
return
"misconfigured"
;
if
(
is_guarded
())
{
/* The instance is managed by Guardian: we can report precise state. */
return
get_instance_state_name
(
get_state
());
}
/* The instance is not managed by Guardian: we can report status only. */
return
is_active
()
?
"online"
:
"offline"
;
}
/**
Reset statistics.
SYNOPSIS
reset_stat()
DESCRIPTION
The operation resets statistics used for guarding the instance.
MT-NOTE: instance must be locked before calling the operation.
TODO: Make private.
*/
void
Instance
::
reset_stat
()
{
restart_counter
=
0
;
crash_moment
=
0
;
last_checked
=
0
;
}
/**************************************************************************
}}}
**************************************************************************/
/**************************************************************************
{{{ Instance: implementation of private operations.
**************************************************************************/
/**
Remove pid file.
*/
...
...
@@ -744,3 +915,36 @@ void Instance::remove_pid()
(
const
char
*
)
options
.
instance_name
.
str
);
}
}
/**
Wait for mysqld to stop within shutdown interval.
*/
bool
Instance
::
wait_for_stop
()
{
int
start_time
=
time
(
NULL
);
int
finish_time
=
start_time
+
options
.
get_shutdown_delay
();
log_info
(
"Instance '%s': waiting for mysqld to stop "
"(timeout: %d seconds)..."
,
(
const
char
*
)
get_name
()
->
str
,
(
int
)
options
.
get_shutdown_delay
());
while
(
true
)
{
if
(
options
.
load_pid
()
==
0
&&
!
is_mysqld_running
())
return
FALSE
;
if
(
time
(
NULL
)
>=
finish_time
)
return
TRUE
;
/* Sleep for 0.3 sec and check again. */
my_sleep
(
300000
);
}
}
/**************************************************************************
}}}
**************************************************************************/
server-tools/instance-manager/instance.h
View file @
d1673c15
...
...
@@ -30,7 +30,7 @@ class Instance_map;
class
Thread_registry
;
/*
/*
*
Instance_name -- the class represents instance name -- a string of length
less than MAX_INSTANCE_NAME_SIZE.
...
...
@@ -68,72 +68,127 @@ private:
class
Instance
{
public:
/*
The following two constants defines name of the default mysqld-instance
("mysqld").
/* States of an instance. */
enum
enum_instance_state
{
STOPPED
,
NOT_STARTED
,
STARTING
,
STARTED
,
JUST_CRASHED
,
CRASHED
,
CRASHED_AND_ABANDONED
,
STOPPING
};
public:
/**
The constant defines name of the default mysqld-instance ("mysqld").
*/
static
const
LEX_STRING
DFLT_INSTANCE_NAME
;
public:
/*
The operation is intended to check whether string is a well-formed
instance name or not.
*/
static
bool
is_name_valid
(
const
LEX_STRING
*
name
);
/*
The operation is intended to check if the given instance name is
mysqld-compatible or not.
*/
static
bool
is_mysqld_compatible_name
(
const
LEX_STRING
*
name
);
public:
Instance
();
~
Instance
();
bool
init
(
const
LEX_STRING
*
name_arg
);
bool
complete_initialization
();
public:
bool
is_active
();
bool
is_mysqld_running
();
int
start
();
int
stop
();
/* send a signal to the instance */
bool
start_mysqld
();
bool
stop_mysqld
();
void
kill_mysqld
(
int
signo
);
bool
is_crashed
();
void
set_crash_flag_n_wake_all
();
/*
void
lock
();
void
unlock
();
const
char
*
get_state_name
();
void
reset_stat
();
public:
/**
The operation is intended to check if the instance is mysqld-compatible
or not.
*/
inline
bool
is_mysqld_compatible
()
const
;
/*
/*
*
The operation is intended to check if the instance is configured properly
or not. Misconfigured instances are not managed.
*/
inline
bool
is_configured
()
const
;
/**
The operation returns TRUE if the instance is guarded and FALSE otherwise.
*/
inline
bool
is_guarded
()
const
;
/**
The operation returns name of the instance.
*/
inline
const
LEX_STRING
*
get_name
()
const
;
/**
The operation returns the current state of the instance.
NOTE: At the moment should be used only for guarded instances.
*/
inline
enum_instance_state
get_state
()
const
;
/**
The operation changes the state of the instance.
NOTE: At the moment should be used only for guarded instances.
TODO: Make private.
*/
inline
void
set_state
(
enum_instance_state
new_state
);
/**
The operation returns crashed flag.
*/
inline
bool
is_crashed
();
public:
enum
{
DEFAULT_SHUTDOWN_DELAY
=
35
};
/**
This attributes contains instance options.
TODO: Make private.
*/
Instance_options
options
;
private:
/* This attributes is a flag, specifies if the instance has been crashed. */
/**
monitoring_thread_active is TRUE if there is a thread that monitors the
corresponding mysqld-process.
*/
bool
monitoring_thread_active
;
/**
crashed is TRUE when corresponding mysqld-process has been died after
start.
*/
bool
crashed
;
/*
This attribute specifies if the instance is configured properly or not
.
/*
*
configured is TRUE when the instance is configured and FALSE otherwise
.
Misconfigured instances are not managed.
*/
bool
configured
;
/*
This attribute specifies whether the instance is mysqld-compatible or not.
Mysqld-compatible instances can contain only mysqld-specific options.
At the moment an instance is mysqld-compatible if its name is "mysqld".
mysqld_compatible specifies whether the instance is mysqld-compatible
or not. Mysqld-compatible instances can contain only mysqld-specific
options. At the moment an instance is mysqld-compatible if its name is
"mysqld".
The idea is that [mysqld] section should contain only mysqld-specific
options (no Instance Manager-specific options) to be readable by mysqld
...
...
@@ -142,18 +197,36 @@ private:
bool
mysqld_compatible
;
/*
Mutex protecting the instance. Currently we use it to avoid the
double start of the instance. This happens when the instance is starting
and we issue the start command once more.
Mutex protecting the instance.
*/
pthread_mutex_t
LOCK_instance
;
/*
This condition variable is used to wake threads waiting for instance to
stop in Instance::stop()
*/
pthread_cond_t
COND_instance_stopped
;
private:
/* Guarded-instance attributes. */
/* state of an instance (i.e. STARTED, CRASHED, etc.) */
enum_instance_state
state
;
public:
/* the amount of attemts to restart instance (cleaned up at success) */
int
restart_counter
;
/* triggered at a crash */
time_t
crash_moment
;
/* General time field. Used to provide timeouts (at shutdown and restart) */
time_t
last_checked
;
private:
static
const
char
*
get_instance_state_name
(
enum_instance_state
state
);
private:
void
remove_pid
();
bool
wait_for_stop
();
private:
friend
class
Instance_monitor
;
};
...
...
@@ -169,9 +242,33 @@ inline bool Instance::is_configured() const
}
inline
bool
Instance
::
is_guarded
()
const
{
return
!
options
.
nonguarded
;
}
inline
const
LEX_STRING
*
Instance
::
get_name
()
const
{
return
&
options
.
instance_name
;
}
inline
Instance
::
enum_instance_state
Instance
::
get_state
()
const
{
return
state
;
}
inline
void
Instance
::
set_state
(
enum_instance_state
new_state
)
{
state
=
new_state
;
}
inline
bool
Instance
::
is_crashed
()
{
return
crashed
;
}
#endif
/* INCLUDES_MYSQL_INSTANCE_MANAGER_INSTANCE_H */
server-tools/instance-manager/instance_map.cc
View file @
d1673c15
...
...
@@ -25,26 +25,18 @@
#include <mysql_com.h>
#include "buffer.h"
#include "guardian.h"
#include "instance.h"
#include "log.h"
#include "manager.h"
#include "mysqld_error.h"
#include "mysql_manager_error.h"
#include "options.h"
#include "priv.h"
/*
Note: As we are going to suppost different types of connections,
we shouldn't have connection-specific functions. To avoid it we could
put such functions to the Command-derived class instead.
The command could be easily constructed for a specific connection if
we would provide a special factory for each connection.
*/
C_MODE_START
/* Procedure needed for HASH initialization */
/**
HASH-routines: get key of instance for storing in hash.
*/
static
byte
*
get_instance_key
(
const
byte
*
u
,
uint
*
len
,
my_bool
__attribute__
((
unused
))
t
)
...
...
@@ -54,14 +46,18 @@ static byte* get_instance_key(const byte* u, uint* len,
return
(
byte
*
)
instance
->
options
.
instance_name
.
str
;
}
/**
HASH-routines: cleanup handler.
*/
static
void
delete_instance
(
void
*
u
)
{
Instance
*
instance
=
(
Instance
*
)
u
;
delete
instance
;
}
/*
The option handler to pass to the process_default_option_files f
i
nction.
/*
*
The option handler to pass to the process_default_option_files f
u
nction.
SYNOPSIS
process_option()
...
...
@@ -96,7 +92,7 @@ static int process_option(void *ctx, const char *group, const char *option)
C_MODE_END
/*
/*
*
Parse option string.
SYNOPSIS
...
...
@@ -137,7 +133,7 @@ static void parse_option(const char *option_str,
}
/*
/*
*
Process one option from the configuration file.
SYNOPSIS
...
...
@@ -151,6 +147,10 @@ static void parse_option(const char *option_str,
process_option(). The caller ensures proper locking
of the instance map object.
*/
/*
Process a given option and assign it to appropricate instance. This is
required for the option handler, passed to my_search_option_files().
*/
int
Instance_map
::
process_one_option
(
const
LEX_STRING
*
group
,
const
char
*
option
)
...
...
@@ -213,92 +213,97 @@ int Instance_map::process_one_option(const LEX_STRING *group,
}
/**
Instance_map constructor.
*/
Instance_map
::
Instance_map
()
{
pthread_mutex_init
(
&
LOCK_instance_map
,
0
);
}
/**
Initialize Instance_map internals.
*/
bool
Instance_map
::
init
()
{
return
hash_init
(
&
hash
,
default_charset_info
,
START_HASH_SIZE
,
0
,
0
,
get_instance_key
,
delete_instance
,
0
);
}
/**
Reset Instance_map data.
*/
bool
Instance_map
::
reset
()
{
hash_free
(
&
hash
);
return
init
();
}
/**
Instance_map destructor.
*/
Instance_map
::~
Instance_map
()
{
pthread_mutex_lock
(
&
LOCK_instance_map
);
lock
();
/*
NOTE: it's necessary to synchronize on each instance before removal,
because Instance-monitoring thread can be still alive an hold the mutex
(because it is detached and we have no control over it).
*/
while
(
true
)
{
Iterator
it
(
this
);
Instance
*
instance
=
it
.
next
();
if
(
!
instance
)
break
;
instance
->
lock
();
instance
->
unlock
();
remove_instance
(
instance
);
}
hash_free
(
&
hash
);
pthread_mutex_unlock
(
&
LOCK_instance_map
);
unlock
();
pthread_mutex_destroy
(
&
LOCK_instance_map
);
}
/**
Lock Instance_map.
*/
void
Instance_map
::
lock
()
{
pthread_mutex_lock
(
&
LOCK_instance_map
);
}
/**
Unlock Instance_map.
*/
void
Instance_map
::
unlock
()
{
pthread_mutex_unlock
(
&
LOCK_instance_map
);
}
/*
Re-read instance configuration file.
SYNOPSIS
Instance_map::flush_instances()
DESCRIPTION
This function will:
- clear the current list of instances. This removes both
running and stopped instances.
- load a new instance configuration from the file.
- pass on the new map to the guardian thread: it will start
all instances that are marked `guarded' and not yet started.
Note, as the check whether an instance is started is currently
very simple (returns TRUE if there is a MySQL server running
at the given port), this function has some peculiar
side-effects:
* if the port number of a running instance was changed, the
old instance is forgotten, even if it was running. The new
instance will be started at the new port.
* if the configuration was changed in a way that two
instances swapped their port numbers, the guardian thread
will not notice that and simply report that both instances
are configured successfully and running.
In order to avoid such side effects one should never call
FLUSH INSTANCES without prior stop of all running instances.
NOTE: The operation should be invoked with the following locks acquired:
- Guardian;
- Instance_map;
/**
Check if there is an active instance or not.
*/
int
Instance_map
::
flush_instances
()
{
int
rc
;
/*
Guardian thread relies on the instance map repository for guarding
instances. This is why refreshing instance map, we need (1) to stop
guardian (2) reload the instance map (3) reinitialize the guardian
with new instances.
*/
hash_free
(
&
hash
);
hash_init
(
&
hash
,
default_charset_info
,
START_HASH_SIZE
,
0
,
0
,
get_instance_key
,
delete_instance
,
0
);
rc
=
load
();
/* don't init guardian if we failed to load instances */
if
(
!
rc
)
guardian
->
init
();
// TODO: check error status.
return
rc
;
}
bool
Instance_map
::
is_there_active_instance
()
{
Instance
*
instance
;
...
...
@@ -306,29 +311,50 @@ bool Instance_map::is_there_active_instance()
while
((
instance
=
iterator
.
next
()))
{
if
(
guardian
->
find_instance_node
(
instance
)
!=
NULL
||
instance
->
is_mysqld_running
())
{
bool
active_instance_found
;
instance
->
lock
();
active_instance_found
=
instance
->
is_active
();
instance
->
unlock
();
if
(
active_instance_found
)
return
TRUE
;
}
}
return
FALSE
;
}
/**
Add an instance into the internal hash.
MT-NOTE: Instance Map must be locked before calling the operation.
*/
int
Instance_map
::
add_instance
(
Instance
*
instance
)
{
return
my_hash_insert
(
&
hash
,
(
byte
*
)
instance
);
}
/**
Remove instance from the internal hash.
MT-NOTE: Instance Map must be locked before calling the operation.
*/
int
Instance_map
::
remove_instance
(
Instance
*
instance
)
{
return
hash_delete
(
&
hash
,
(
byte
*
)
instance
);
}
/**
Create a new instance and register it in the internal hash.
MT-NOTE: Instance Map must be locked before calling the operation.
*/
int
Instance_map
::
create_instance
(
const
LEX_STRING
*
instance_name
,
const
Named_value_arr
*
options
)
{
...
...
@@ -392,12 +418,22 @@ int Instance_map::create_instance(const LEX_STRING *instance_name,
}
/**
Return a pointer to the instance or NULL, if there is no such instance.
MT-NOTE: Instance Map must be locked before calling the operation.
*/
Instance
*
Instance_map
::
find
(
const
LEX_STRING
*
name
)
{
return
(
Instance
*
)
hash_search
(
&
hash
,
(
byte
*
)
name
->
str
,
name
->
length
);
}
/**
Init instances command line arguments after all options have been loaded.
*/
bool
Instance_map
::
complete_initialization
()
{
bool
mysqld_found
;
...
...
@@ -455,7 +491,10 @@ bool Instance_map::complete_initialization()
}
/* load options from config files and create appropriate instance structures */
/**
Load options from config files and create appropriate instance
structures.
*/
int
Instance_map
::
load
()
{
...
...
@@ -505,8 +544,9 @@ int Instance_map::load()
}
/*--- Implementaton of the Instance map iterator class ---*/
/*************************************************************************
{{{ Instance_map::Iterator implementation.
*************************************************************************/
void
Instance_map
::
Iterator
::
go_to_first
()
{
...
...
@@ -522,29 +562,12 @@ Instance *Instance_map::Iterator::next()
return
NULL
;
}
const
char
*
Instance_map
::
get_instance_state_name
(
Instance
*
instance
)
{
LIST
*
instance_node
;
if
(
!
instance
->
is_configured
())
return
"misconfigured"
;
if
((
instance_node
=
guardian
->
find_instance_node
(
instance
))
!=
NULL
)
{
/* The instance is managed by Guardian: we can report precise state. */
return
Guardian
::
get_instance_state_name
(
guardian
->
get_instance_state
(
instance_node
));
}
/* The instance is not managed by Guardian: we can report status only. */
return
instance
->
is_mysqld_running
()
?
"online"
:
"offline"
;
}
/*************************************************************************
}}}
*************************************************************************/
/*
/*
*
Create a new configuration section for mysqld-instance in the config file.
SYNOPSIS
...
...
server-tools/instance-manager/instance_map.h
View file @
d1673c15
...
...
@@ -37,14 +37,17 @@ extern int create_instance_in_file(const LEX_STRING *instance_name,
const
Named_value_arr
*
options
);
/*
/*
*
Instance_map - stores all existing instances
*/
class
Instance_map
{
public:
/* Instance_map iterator */
/**
Instance_map iterator
*/
class
Iterator
{
private:
...
...
@@ -58,79 +61,43 @@ public:
void
go_to_first
();
Instance
*
next
();
};
friend
class
Iterator
;
public:
/*
Return a pointer to the instance or NULL, if there is no such instance.
MT-NOTE: must be called under acquired lock.
*/
Instance
*
find
(
const
LEX_STRING
*
name
);
/* Clear the configuration cache and reload the configuration file. */
int
flush_instances
();
/* The operation is used to check if there is an active instance or not. */
bool
is_there_active_instance
();
void
lock
();
void
unlock
();
bool
init
();
bool
reset
();
/*
Process a given option and assign it to appropricate instance. This is
required for the option handler, passed to my_search_option_files().
*/
int
process_one_option
(
const
LEX_STRING
*
group
,
const
char
*
option
);
int
load
();
/*
Add an instance into the internal hash.
int
process_one_option
(
const
LEX_STRING
*
group
,
const
char
*
option
);
MT-NOTE: the operation must be called under acquired lock.
*/
int
add_instance
(
Instance
*
instance
);
/*
Remove instance from the internal hash.
MT-NOTE: the operation must be called under acquired lock.
*/
int
remove_instance
(
Instance
*
instance
);
/*
Create a new instance and register it in the internal hash.
MT-NOTE: the operation must be called under acquired lock.
*/
int
create_instance
(
const
LEX_STRING
*
instance_name
,
const
Named_value_arr
*
options
);
public:
Instance_map
();
~
Instance_map
();
/*
Retrieve client state name of the given instance.
MT-NOTE: the options must be called under acquired locks of the following
objects:
- Instance_map;
- Guardian;
*/
const
char
*
get_instance_state_name
(
Instance
*
instance
);
public:
const
char
*
mysqld_path
;
Guardian
*
guardian
;
private:
/* loads options from config files */
int
load
();
/* inits instances argv's after all options have been loaded */
bool
complete_initialization
();
private:
enum
{
START_HASH_SIZE
=
16
};
pthread_mutex_t
LOCK_instance_map
;
HASH
hash
;
private:
friend
class
Iterator
;
};
#endif
/* INCLUDES_MYSQL_INSTANCE_MANAGER_INSTANCE_MAP_H */
server-tools/instance-manager/instance_options.h
View file @
d1673c15
...
...
@@ -46,7 +46,6 @@ public:
Instance_options
();
~
Instance_options
();
/* fills in argv */
bool
complete_initialization
();
bool
set_option
(
Named_value
*
option
);
...
...
server-tools/instance-manager/manager.cc
View file @
d1673c15
...
...
@@ -37,6 +37,9 @@
#include "user_map.h"
/**********************************************************************
{{{ Platform-specific implementation.
**********************************************************************/
#ifndef __WIN__
void
set_signals
(
sigset_t
*
mask
)
...
...
@@ -92,9 +95,13 @@ int my_sigwait(const sigset_t *set, int *sig)
#endif
/**********************************************************************
}}}
**********************************************************************/
/**********************************************************************
Implementation of checking the actual thread model.
{{{
Implementation of checking the actual thread model.
***********************************************************************/
namespace
{
/* no-indent */
...
...
@@ -137,6 +144,10 @@ bool check_if_linux_threads(bool *linux_threads)
}
/**********************************************************************
}}}
***********************************************************************/
/**********************************************************************
Manager implementation
...
...
@@ -152,23 +163,35 @@ bool Manager::linux_threads;
#endif // __WIN__
/**
Request shutdown of guardian and threads registered in Thread_registry.
SYNOPSIS
stop_all_threads()
*/
void
Manager
::
stop_all_threads
()
{
/*
Let
g
uardian thread know that it should break it's processing cycle,
Let
G
uardian thread know that it should break it's processing cycle,
once it wakes up.
*/
p_guardian
->
request_shutdown
();
/* wake guardian */
pthread_cond_signal
(
&
p_guardian
->
COND_guardian
);
/* stop all threads */
/* Stop all threads. */
p_thread_registry
->
deliver_shutdown
();
}
/*
manager - entry point to the main instance manager process: start
listener thread, write pid file and enter into signal handling.
/**
Main manager function.
SYNOPSIS
main()
DESCRIPTION
This is an entry point to the main instance manager process:
start listener thread, write pid file and enter into signal handling.
See also comments in mysqlmanager.cc to picture general Instance Manager
architecture.
...
...
@@ -194,22 +217,33 @@ int Manager::main()
(
const
char
*
)
(
linux_threads
?
"LINUX threads"
:
"POSIX threads"
));
#endif // __WIN__
Thread_registry
thread_registry
;
/*
All objects created in the manager() function live as long as
thread_registry lives, and thread_registry is alive until there are
working threads.
All objects created in the Manager object live as long as thread_registry
lives, and thread_registry is alive until there are working threads.
There are two main purposes of the Thread Registry:
1. Interrupt blocking I/O and signal condition variables in case of
shutdown;
2. Wait for detached threads before shutting down the main thread.
NOTE:
1. Handling shutdown can be done in more elegant manner by introducing
Event (or Condition) object with support of logical operations.
2. Using Thread Registry to wait for detached threads is definitely not
the best way, because when Thread Registry unregisters an thread, the
thread is still alive. Accurate way to wait for threads to stop is
not using detached threads and join all threads before shutdown.
*/
Thread_registry
thread_registry
;
User_map
user_map
;
Instance_map
instance_map
;
Guardian
guardian
(
&
thread_registry
,
&
instance_map
,
Options
::
Main
::
monitoring_interval
);
Guardian
guardian
(
&
thread_registry
,
&
instance_map
);
Listener
listener
(
&
thread_registry
,
&
user_map
);
p_instance_map
=
&
instance_map
;
p_guardian
=
instance_map
.
guardian
=
&
guardian
;
p_guardian
=
&
guardian
;
p_thread_registry
=
&
thread_registry
;
p_user_map
=
&
user_map
;
...
...
@@ -249,7 +283,7 @@ int Manager::main()
}
}
/*
write Instance Manager pid file
*/
/*
Write Instance Manager pid file.
*/
log_info
(
"IM pid file: '%s'; PID: %d."
,
(
const
char
*
)
Options
::
Main
::
pid_file_name
,
...
...
@@ -290,6 +324,7 @@ int Manager::main()
permitted to process instances. And before flush_instances() has
completed, there are no instances to guard.
*/
if
(
guardian
.
start
(
Thread
::
DETACHED
))
{
log_error
(
"Can not start Guardian thread."
);
...
...
@@ -298,22 +333,12 @@ int Manager::main()
/* Load instances. */
{
instance_map
.
guardian
->
lock
();
instance_map
.
lock
();
int
flush_instances_status
=
instance_map
.
flush_instances
();
instance_map
.
unlock
();
instance_map
.
guardian
->
unlock
();
if
(
flush_instances_status
)
if
(
Manager
::
flush_instances
())
{
log_error
(
"Can not init instances repository."
);
stop_all_threads
();
goto
err
;
}
}
/* Initialize the Listener. */
...
...
@@ -328,7 +353,8 @@ int Manager::main()
After the list of guarded instances have been initialized,
Guardian should start them.
*/
pthread_cond_signal
(
&
guardian
.
COND_guardian
);
guardian
.
ping
();
/* Main loop. */
...
...
@@ -381,7 +407,6 @@ int Manager::main()
if
(
!
guardian
.
is_stopped
())
{
guardian
.
request_shutdown
();
pthread_cond_signal
(
&
guardian
.
COND_guardian
);
}
else
{
...
...
@@ -406,3 +431,64 @@ err:
#endif
return
rc
;
}
/**
Re-read instance configuration file.
SYNOPSIS
flush_instances()
DESCRIPTION
This function will:
- clear the current list of instances. This removes both
running and stopped instances.
- load a new instance configuration from the file.
- pass on the new map to the guardian thread: it will start
all instances that are marked `guarded' and not yet started.
Note, as the check whether an instance is started is currently
very simple (returns TRUE if there is a MySQL server running
at the given port), this function has some peculiar
side-effects:
* if the port number of a running instance was changed, the
old instance is forgotten, even if it was running. The new
instance will be started at the new port.
* if the configuration was changed in a way that two
instances swapped their port numbers, the guardian thread
will not notice that and simply report that both instances
are configured successfully and running.
In order to avoid such side effects one should never call
FLUSH INSTANCES without prior stop of all running instances.
*/
bool
Manager
::
flush_instances
()
{
p_instance_map
->
lock
();
if
(
p_instance_map
->
is_there_active_instance
())
{
p_instance_map
->
unlock
();
return
TRUE
;
}
if
(
p_instance_map
->
reset
())
{
p_instance_map
->
unlock
();
return
TRUE
;
}
if
(
p_instance_map
->
load
())
{
p_instance_map
->
unlock
();
return
TRUE
;
/* Don't init guardian if we failed to load instances. */
}
get_guardian
()
->
init
();
/* TODO: check error status. */
get_guardian
()
->
ping
();
p_instance_map
->
unlock
();
return
FALSE
;
}
server-tools/instance-manager/manager.h
View file @
d1673c15
...
...
@@ -19,6 +19,7 @@
#if defined(__GNUC__) && defined(USE_PRAGMA_INTERFACE)
#pragma interface
#endif
#include <my_global.h>
class
Guardian
;
...
...
@@ -30,8 +31,12 @@ class Manager
{
public:
static
int
main
();
static
bool
flush_instances
();
public:
/**
These methods return a non-
zero
value only for the duration
These methods return a non-
NULL
value only for the duration
of main().
*/
static
Instance_map
*
get_instance_map
()
{
return
p_instance_map
;
}
...
...
@@ -39,6 +44,7 @@ public:
static
Thread_registry
*
get_thread_registry
()
{
return
p_thread_registry
;
}
static
User_map
*
get_user_map
()
{
return
p_user_map
;
}
public:
#ifndef __WIN__
static
bool
is_linux_threads
()
{
return
linux_threads
;
}
#endif // __WIN__
...
...
server-tools/instance-manager/user_map.cc
View file @
d1673c15
...
...
@@ -78,7 +78,7 @@ int User::init(const char *line)
"User line: '%s'."
,
(
int
)
password_length
,
(
int
)
SCRAMBLED_PASSWORD_CHAR_LENGTH
,
line
);
(
const
char
*
)
line
);
return
1
;
}
...
...
sql/handler.cc
View file @
d1673c15
...
...
@@ -692,6 +692,19 @@ int ha_commit_trans(THD *thd, bool all)
ha_rollback_trans
(
thd
,
all
);
DBUG_RETURN
(
1
);
}
if
(
is_real_trans
&&
opt_readonly
&&
!
(
thd
->
security_ctx
->
master_access
&
SUPER_ACL
)
&&
!
thd
->
slave_thread
)
{
my_error
(
ER_OPTION_PREVENTS_STATEMENT
,
MYF
(
0
),
"--read-only"
);
ha_rollback_trans
(
thd
,
all
);
error
=
1
;
goto
end
;
}
DBUG_EXECUTE_IF
(
"crash_commit_before"
,
abort
(););
/* Close all cursors that can not survive COMMIT */
...
...
sql/lock.cc
View file @
d1673c15
...
...
@@ -151,6 +151,23 @@ MYSQL_LOCK *mysql_lock_tables(THD *thd, TABLE **tables, uint count,
}
}
if
(
write_lock_used
&&
opt_readonly
&&
!
(
thd
->
security_ctx
->
master_access
&
SUPER_ACL
)
&&
!
thd
->
slave_thread
)
{
/*
Someone has issued SET GLOBAL READ_ONLY=1 and we want a write lock.
We do not wait for READ_ONLY=0, and fail.
*/
reset_lock_data
(
sql_lock
);
my_free
((
gptr
)
sql_lock
,
MYF
(
0
));
sql_lock
=
0
;
my_error
(
ER_OPTION_PREVENTS_STATEMENT
,
MYF
(
0
),
"--read-only"
);
break
;
}
thd
->
proc_info
=
"System lock"
;
DBUG_PRINT
(
"info"
,
(
"thd->proc_info %s"
,
thd
->
proc_info
));
if
(
lock_external
(
thd
,
tables
,
count
))
...
...
sql/set_var.cc
View file @
d1673c15
...
...
@@ -369,7 +369,7 @@ sys_var_thd_ulong sys_preload_buff_size("preload_buffer_size",
&
SV
::
preload_buff_size
);
sys_var_thd_ulong
sys_read_buff_size
(
"read_buffer_size"
,
&
SV
::
read_buff_size
);
sys_var_
bool_ptr
sys_readonly
(
"read_only"
,
&
opt_readonly
);
sys_var_
opt_readonly
sys_readonly
(
"read_only"
,
&
opt_readonly
);
sys_var_thd_ulong
sys_read_rnd_buff_size
(
"read_rnd_buffer_size"
,
&
SV
::
read_rnd_buff_size
);
sys_var_thd_ulong
sys_div_precincrement
(
"div_precision_increment"
,
...
...
@@ -3856,6 +3856,70 @@ bool sys_var_trust_routine_creators::update(THD *thd, set_var *var)
return
sys_var_bool_ptr
::
update
(
thd
,
var
);
}
bool
sys_var_opt_readonly
::
update
(
THD
*
thd
,
set_var
*
var
)
{
bool
result
;
DBUG_ENTER
(
"sys_var_opt_readonly::update"
);
/* Prevent self dead-lock */
if
(
thd
->
locked_tables
||
thd
->
active_transaction
())
{
my_error
(
ER_LOCK_OR_ACTIVE_TRANSACTION
,
MYF
(
0
));
DBUG_RETURN
(
true
);
}
if
(
thd
->
global_read_lock
)
{
/*
This connection already holds the global read lock.
This can be the case with:
- FLUSH TABLES WITH READ LOCK
- SET GLOBAL READ_ONLY = 1
*/
result
=
sys_var_bool_ptr
::
update
(
thd
,
var
);
DBUG_RETURN
(
result
);
}
/*
Perform a 'FLUSH TABLES WITH READ LOCK'.
This is a 3 step process:
- [1] lock_global_read_lock()
- [2] close_cached_tables()
- [3] make_global_read_lock_block_commit()
[1] prevents new connections from obtaining tables locked for write.
[2] waits until all existing connections close their tables.
[3] prevents transactions from being committed.
*/
if
(
lock_global_read_lock
(
thd
))
DBUG_RETURN
(
true
);
/*
This call will be blocked by any connection holding a READ or WRITE lock.
Ideally, we want to wait only for pending WRITE locks, but since:
con 1> LOCK TABLE T FOR READ;
con 2> LOCK TABLE T FOR WRITE; (blocked by con 1)
con 3> SET GLOBAL READ ONLY=1; (blocked by con 2)
can cause to wait on a read lock, it's required for the client application
to unlock everything, and acceptable for the server to wait on all locks.
*/
if
(
close_cached_tables
(
thd
,
true
,
NULL
,
false
))
goto
end_with_read_lock
;
if
(
result
=
make_global_read_lock_block_commit
(
thd
))
goto
end_with_read_lock
;
/* Change the opt_readonly system variable, safe because the lock is held */
result
=
sys_var_bool_ptr
::
update
(
thd
,
var
);
end_with_read_lock:
/* Release the lock */
unlock_global_read_lock
(
thd
);
DBUG_RETURN
(
result
);
}
/* even session variable here requires SUPER, because of -#o,file */
bool
sys_var_thd_dbug
::
check
(
THD
*
thd
,
set_var
*
var
)
{
...
...
sql/set_var.h
View file @
d1673c15
...
...
@@ -905,6 +905,20 @@ public:
};
/**
Handler for setting the system variable --read-only.
*/
class
sys_var_opt_readonly
:
public
sys_var_bool_ptr
{
public:
sys_var_opt_readonly
(
const
char
*
name_arg
,
my_bool
*
value_arg
)
:
sys_var_bool_ptr
(
name_arg
,
value_arg
)
{};
~
sys_var_opt_readonly
()
{};
bool
update
(
THD
*
thd
,
set_var
*
var
);
};
class
sys_var_thd_lc_time_names
:
public
sys_var_thd
{
public:
...
...
sql/sql_class.cc
View file @
d1673c15
...
...
@@ -985,6 +985,13 @@ void select_result::cleanup()
/* do nothing */
}
bool
select_result
::
check_simple_select
()
const
{
my_error
(
ER_SP_BAD_CURSOR_QUERY
,
MYF
(
0
));
return
TRUE
;
}
static
String
default_line_term
(
"
\n
"
,
default_charset_info
);
static
String
default_escaped
(
"
\\
"
,
default_charset_info
);
static
String
default_field_term
(
"
\t
"
,
default_charset_info
);
...
...
@@ -1656,6 +1663,13 @@ int select_dumpvar::prepare(List<Item> &list, SELECT_LEX_UNIT *u)
}
bool
select_dumpvar
::
check_simple_select
()
const
{
my_error
(
ER_SP_BAD_CURSOR_SELECT
,
MYF
(
0
));
return
TRUE
;
}
void
select_dumpvar
::
cleanup
()
{
vars
.
empty
();
...
...
sql/sql_class.h
View file @
d1673c15
...
...
@@ -1703,7 +1703,14 @@ public:
virtual
bool
initialize_tables
(
JOIN
*
join
=
0
)
{
return
0
;
}
virtual
void
send_error
(
uint
errcode
,
const
char
*
err
);
virtual
bool
send_eof
()
=
0
;
virtual
bool
simple_select
()
{
return
0
;
}
/**
Check if this query returns a result set and therefore is allowed in
cursors and set an error message if it is not the case.
@retval FALSE success
@retval TRUE error, an error message is set
*/
virtual
bool
check_simple_select
()
const
;
virtual
void
abort
()
{}
/*
Cleanup instance of this class for next execution of a prepared
...
...
@@ -1741,7 +1748,7 @@ public:
bool
send_fields
(
List
<
Item
>
&
list
,
uint
flags
);
bool
send_data
(
List
<
Item
>
&
items
);
bool
send_eof
();
bool
simple_select
()
{
return
1
;
}
virtual
bool
check_simple_select
()
const
{
return
FALSE
;
}
void
abort
();
};
...
...
@@ -2189,6 +2196,7 @@ public:
int
prepare
(
List
<
Item
>
&
list
,
SELECT_LEX_UNIT
*
u
);
bool
send_data
(
List
<
Item
>
&
items
);
bool
send_eof
();
virtual
bool
check_simple_select
()
const
;
void
cleanup
();
};
...
...
sql/sql_prepare.cc
View file @
d1673c15
...
...
@@ -2946,10 +2946,9 @@ bool Prepared_statement::execute(String *expanded_query, bool open_cursor)
in INSERT ... SELECT and similar commands.
*/
if
(
open_cursor
&&
lex
->
result
&&
!
lex
->
result
->
simple_select
())
if
(
open_cursor
&&
lex
->
result
&&
lex
->
result
->
check_
simple_select
())
{
DBUG_PRINT
(
"info"
,(
"Cursor asked for not SELECT stmt"
));
my_error
(
ER_SP_BAD_CURSOR_QUERY
,
MYF
(
0
));
return
TRUE
;
}
...
...
tests/mysql_client_test.c
View file @
d1673c15
...
...
@@ -15684,6 +15684,33 @@ static void test_bug21635()
DBUG_VOID_RETURN
;
}
/*
Bug#24179 "select b into $var" fails with --cursor_protocol"
The failure is correct, check that the returned message is meaningful.
*/
static
void
test_bug24179
()
{
int
rc
;
MYSQL_STMT
*
stmt
;
DBUG_ENTER
(
"test_bug24179"
);
myheader
(
"test_bug24179"
);
stmt
=
open_cursor
(
"select 1 into @a"
);
rc
=
mysql_stmt_execute
(
stmt
);
DIE_UNLESS
(
rc
);
if
(
!
opt_silent
)
{
printf
(
"Got error (as expected): %d %s
\n
"
,
mysql_stmt_errno
(
stmt
),
mysql_stmt_error
(
stmt
));
}
DIE_UNLESS
(
mysql_stmt_errno
(
stmt
)
==
1323
);
DBUG_VOID_RETURN
;
}
/*
Read and parse arguments and MySQL options from my.cnf
...
...
@@ -15966,6 +15993,7 @@ static struct my_tests_st my_tests[]= {
{
"test_bug23383"
,
test_bug23383
},
{
"test_bug21635"
,
test_bug21635
},
{
"test_status"
,
test_status
},
{
"test_bug24179"
,
test_bug24179
},
{
0
,
0
}
};
...
...
Write
Preview
Markdown
is supported
0%
Try again
or
attach a new file
Attach a file
Cancel
You are about to add
0
people
to the discussion. Proceed with caution.
Finish editing this message first!
Cancel
Please
register
or
sign in
to comment