2007-05-18 14:06:29 +04:00
/*
monitoring links to all other nodes to detect dead nodes
Copyright ( C ) Ronnie Sahlberg 2007
2007-05-31 07:50:53 +04:00
This program is free software ; you can redistribute it and / or modify
it under the terms of the GNU General Public License as published by
2007-07-10 09:29:31 +04:00
the Free Software Foundation ; either version 3 of the License , or
2007-05-31 07:50:53 +04:00
( at your option ) any later version .
This program is distributed in the hope that it will be useful ,
2007-05-18 14:06:29 +04:00
but WITHOUT ANY WARRANTY ; without even the implied warranty of
2007-05-31 07:50:53 +04:00
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE . See the
GNU General Public License for more details .
You should have received a copy of the GNU General Public License
2007-07-10 09:29:31 +04:00
along with this program ; if not , see < http : //www.gnu.org/licenses/>.
2007-05-18 14:06:29 +04:00
*/
# include "includes.h"
# include "lib/events/events.h"
# include "system/filesys.h"
# include "system/wait.h"
# include "../include/ctdb_private.h"
2008-01-10 06:40:56 +03:00
struct ctdb_monitor_state {
uint32_t monitoring_mode ;
TALLOC_CTX * monitor_context ;
uint32_t next_interval ;
} ;
2007-06-06 04:25:46 +04:00
static void ctdb_check_health ( struct event_context * ev , struct timed_event * te ,
struct timeval t , void * private_data ) ;
2009-03-31 07:23:31 +04:00
/*
setup the notification script
*/
int ctdb_set_notification_script ( struct ctdb_context * ctdb , const char * script )
{
ctdb - > notification_script = talloc_strdup ( ctdb , script ) ;
CTDB_NO_MEMORY ( ctdb , ctdb - > notification_script ) ;
return 0 ;
}
static int ctdb_run_notification_script_child ( struct ctdb_context * ctdb , const char * event )
{
struct stat st ;
int ret ;
char * cmd ;
if ( stat ( ctdb - > notification_script , & st ) ! = 0 ) {
DEBUG ( DEBUG_ERR , ( " Could not stat notification script %s. Can not send notifications. \n " , ctdb - > notification_script ) ) ;
return - 1 ;
}
if ( ! ( st . st_mode & S_IXUSR ) ) {
DEBUG ( DEBUG_ERR , ( " Notification script %s is not executable. \n " , ctdb - > notification_script ) ) ;
return - 1 ;
}
cmd = talloc_asprintf ( ctdb , " %s %s \n " , ctdb - > notification_script , event ) ;
CTDB_NO_MEMORY ( ctdb , cmd ) ;
ret = system ( cmd ) ;
/* if the system() call was successful, translate ret into the
return code from the command
*/
if ( ret ! = - 1 ) {
ret = WEXITSTATUS ( ret ) ;
}
if ( ret ! = 0 ) {
DEBUG ( DEBUG_ERR , ( " Notification script \" %s \" failed with error %d \n " , cmd , ret ) ) ;
}
return ret ;
}
static void ctdb_run_notification_script ( struct ctdb_context * ctdb , const char * event )
{
pid_t child ;
if ( ctdb - > notification_script = = NULL ) {
return ;
}
child = fork ( ) ;
if ( child = = ( pid_t ) - 1 ) {
DEBUG ( DEBUG_ERR , ( " Failed to fork() a notification child process \n " ) ) ;
return ;
}
if ( child = = 0 ) {
int ret ;
ret = ctdb_run_notification_script_child ( ctdb , event ) ;
if ( ret ! = 0 ) {
DEBUG ( DEBUG_ERR , ( __location__ " Notification script failed \n " ) ) ;
}
_exit ( 0 ) ;
}
return ;
}
2007-06-06 04:25:46 +04:00
/*
called when a health monitoring event script finishes
*/
static void ctdb_health_callback ( struct ctdb_context * ctdb , int status , void * p )
{
2007-09-04 04:06:36 +04:00
struct ctdb_node * node = ctdb - > nodes [ ctdb - > pnn ] ;
2007-06-06 04:25:46 +04:00
TDB_DATA data ;
struct ctdb_node_flag_change c ;
2007-09-24 04:12:18 +04:00
uint32_t next_interval ;
2007-06-06 04:25:46 +04:00
2007-09-04 04:33:10 +04:00
c . pnn = ctdb - > pnn ;
2007-08-21 11:25:15 +04:00
c . old_flags = node - > flags ;
2007-06-07 05:15:22 +04:00
if ( status ! = 0 & & ! ( node - > flags & NODE_FLAGS_UNHEALTHY ) ) {
2008-02-04 12:07:15 +03:00
DEBUG ( DEBUG_NOTICE , ( " monitor event failed - disabling node \n " ) ) ;
2007-06-07 05:15:22 +04:00
node - > flags | = NODE_FLAGS_UNHEALTHY ;
2008-01-10 06:40:56 +03:00
ctdb - > monitor - > next_interval = 1 ;
2008-02-22 02:33:09 +03:00
if ( ctdb - > tunable . disable_when_unhealthy ! = 0 ) {
DEBUG ( DEBUG_INFO , ( " DISABLING node since it became unhealthy \n " ) ) ;
node - > flags | = NODE_FLAGS_DISABLED ;
}
2009-03-31 07:23:31 +04:00
ctdb_run_notification_script ( ctdb , " unhealthy " ) ;
2007-06-07 05:15:22 +04:00
} else if ( status = = 0 & & ( node - > flags & NODE_FLAGS_UNHEALTHY ) ) {
2008-02-04 09:44:24 +03:00
DEBUG ( DEBUG_NOTICE , ( " monitor event OK - node re-enabled \n " ) ) ;
2007-09-24 04:12:18 +04:00
node - > flags & = ~ NODE_FLAGS_UNHEALTHY ;
2008-01-10 06:40:56 +03:00
ctdb - > monitor - > next_interval = 1 ;
2009-03-31 07:23:31 +04:00
ctdb_run_notification_script ( ctdb , " healthy " ) ;
2007-09-24 04:12:18 +04:00
}
2008-01-10 06:40:56 +03:00
next_interval = ctdb - > monitor - > next_interval ;
ctdb - > monitor - > next_interval * = 2 ;
if ( ctdb - > monitor - > next_interval > ctdb - > tunable . monitor_interval ) {
ctdb - > monitor - > next_interval = ctdb - > tunable . monitor_interval ;
2007-09-24 04:12:18 +04:00
}
2008-01-10 06:40:56 +03:00
event_add_timed ( ctdb - > ev , ctdb - > monitor - > monitor_context ,
2007-11-30 01:02:37 +03:00
timeval_current_ofs ( next_interval , 0 ) ,
ctdb_check_health , ctdb ) ;
2007-09-24 04:12:18 +04:00
if ( c . old_flags = = node - > flags ) {
2007-06-06 04:25:46 +04:00
return ;
}
2007-08-21 11:25:15 +04:00
c . new_flags = node - > flags ;
2007-06-06 04:25:46 +04:00
data . dptr = ( uint8_t * ) & c ;
data . dsize = sizeof ( c ) ;
2008-11-19 06:43:46 +03:00
/* ask the recovery daemon to push these changes out to all nodes */
ctdb_daemon_send_message ( ctdb , ctdb - > pnn ,
CTDB_SRVID_PUSH_NODE_FLAGS , data ) ;
2007-06-07 05:15:22 +04:00
2007-06-06 04:25:46 +04:00
}
2007-11-12 02:53:11 +03:00
/*
called when the startup event script finishes
*/
static void ctdb_startup_callback ( struct ctdb_context * ctdb , int status , void * p )
{
if ( status ! = 0 ) {
2008-02-04 12:07:15 +03:00
DEBUG ( DEBUG_ERR , ( " startup event failed \n " ) ) ;
2007-11-12 02:53:11 +03:00
} else if ( status = = 0 ) {
2008-02-04 12:07:15 +03:00
DEBUG ( DEBUG_NOTICE , ( " startup event OK - enabling monitoring \n " ) ) ;
2007-11-12 02:53:11 +03:00
ctdb - > done_startup = true ;
2008-01-10 06:40:56 +03:00
ctdb - > monitor - > next_interval = 1 ;
2007-11-12 02:53:11 +03:00
}
2008-01-10 06:40:56 +03:00
event_add_timed ( ctdb - > ev , ctdb - > monitor - > monitor_context ,
timeval_current_ofs ( ctdb - > monitor - > next_interval , 0 ) ,
ctdb_check_health , ctdb ) ;
2007-11-12 02:53:11 +03:00
}
2007-06-06 04:25:46 +04:00
/*
see if the event scripts think we are healthy
*/
static void ctdb_check_health ( struct event_context * ev , struct timed_event * te ,
struct timeval t , void * private_data )
{
struct ctdb_context * ctdb = talloc_get_type ( private_data , struct ctdb_context ) ;
int ret ;
2007-11-12 05:10:15 +03:00
if ( ctdb - > recovery_mode ! = CTDB_RECOVERY_NORMAL | |
2008-01-10 06:40:56 +03:00
( ctdb - > monitor - > monitoring_mode = = CTDB_MONITORING_DISABLED & & ctdb - > done_startup ) ) {
event_add_timed ( ctdb - > ev , ctdb - > monitor - > monitor_context ,
timeval_current_ofs ( ctdb - > monitor - > next_interval , 0 ) ,
2007-06-06 04:25:46 +04:00
ctdb_check_health , ctdb ) ;
return ;
}
2007-11-12 02:53:11 +03:00
if ( ! ctdb - > done_startup ) {
ret = ctdb_event_script_callback ( ctdb ,
timeval_current_ofs ( ctdb - > tunable . script_timeout , 0 ) ,
2008-01-10 06:40:56 +03:00
ctdb - > monitor - > monitor_context , ctdb_startup_callback ,
2007-11-12 02:53:11 +03:00
ctdb , " startup " ) ;
} else {
ret = ctdb_event_script_callback ( ctdb ,
timeval_current_ofs ( ctdb - > tunable . script_timeout , 0 ) ,
2008-01-10 06:40:56 +03:00
ctdb - > monitor - > monitor_context , ctdb_health_callback ,
2007-11-12 02:53:11 +03:00
ctdb , " monitor " ) ;
}
2007-06-06 04:25:46 +04:00
if ( ret ! = 0 ) {
2008-02-04 12:07:15 +03:00
DEBUG ( DEBUG_ERR , ( " Unable to launch monitor event script \n " ) ) ;
2008-01-10 06:49:43 +03:00
ctdb - > monitor - > next_interval = 1 ;
2008-01-10 06:40:56 +03:00
event_add_timed ( ctdb - > ev , ctdb - > monitor - > monitor_context ,
2008-01-10 06:49:43 +03:00
timeval_current_ofs ( 1 , 0 ) ,
2007-06-06 04:25:46 +04:00
ctdb_check_health , ctdb ) ;
}
}
2007-11-30 02:09:54 +03:00
/*
( Temporaily ) Disabling monitoring will stop the monitor event scripts
from running but node health checks will still occur
*/
void ctdb_disable_monitoring ( struct ctdb_context * ctdb )
{
2008-01-10 06:40:56 +03:00
ctdb - > monitor - > monitoring_mode = CTDB_MONITORING_DISABLED ;
2008-02-04 09:44:24 +03:00
DEBUG ( DEBUG_INFO , ( " Monitoring has been disabled \n " ) ) ;
2007-11-30 02:09:54 +03:00
}
/*
Re - enable running monitor events after they have been disabled
*/
void ctdb_enable_monitoring ( struct ctdb_context * ctdb )
{
2008-01-10 06:40:56 +03:00
ctdb - > monitor - > monitoring_mode = CTDB_MONITORING_ACTIVE ;
2008-05-16 02:20:40 +04:00
ctdb - > monitor - > next_interval = 2 ;
2008-02-04 09:44:24 +03:00
DEBUG ( DEBUG_INFO , ( " Monitoring has been enabled \n " ) ) ;
2007-11-30 02:09:54 +03:00
}
/* stop any monitoring
this should only be done when shutting down the daemon
*/
2007-06-06 07:45:12 +04:00
void ctdb_stop_monitoring ( struct ctdb_context * ctdb )
{
2008-01-10 06:40:56 +03:00
talloc_free ( ctdb - > monitor - > monitor_context ) ;
ctdb - > monitor - > monitor_context = NULL ;
2007-11-30 00:44:34 +03:00
2008-01-10 06:40:56 +03:00
ctdb - > monitor - > monitoring_mode = CTDB_MONITORING_DISABLED ;
ctdb - > monitor - > next_interval = 1 ;
2008-02-04 12:07:15 +03:00
DEBUG ( DEBUG_NOTICE , ( " Monitoring has been stopped \n " ) ) ;
2007-06-06 07:45:12 +04:00
}
2007-06-06 04:25:46 +04:00
2007-05-18 17:23:36 +04:00
/*
start watching for nodes that might be dead
*/
2007-06-06 07:45:12 +04:00
void ctdb_start_monitoring ( struct ctdb_context * ctdb )
2007-05-18 14:06:29 +04:00
{
2007-06-06 07:45:12 +04:00
struct timed_event * te ;
2008-01-10 06:40:56 +03:00
if ( ctdb - > monitor ! = NULL ) {
2007-11-30 00:44:34 +03:00
return ;
}
2008-01-10 06:40:56 +03:00
ctdb - > monitor = talloc ( ctdb , struct ctdb_monitor_state ) ;
CTDB_NO_MEMORY_FATAL ( ctdb , ctdb - > monitor ) ;
ctdb - > monitor - > next_interval = 1 ;
2007-06-06 07:45:12 +04:00
2008-01-10 06:40:56 +03:00
ctdb - > monitor - > monitor_context = talloc_new ( ctdb - > monitor ) ;
CTDB_NO_MEMORY_FATAL ( ctdb , ctdb - > monitor - > monitor_context ) ;
2007-11-30 00:44:34 +03:00
2008-01-10 06:40:56 +03:00
te = event_add_timed ( ctdb - > ev , ctdb - > monitor - > monitor_context ,
2008-01-10 06:49:43 +03:00
timeval_current_ofs ( 1 , 0 ) ,
2007-06-06 07:45:12 +04:00
ctdb_check_health , ctdb ) ;
CTDB_NO_MEMORY_FATAL ( ctdb , te ) ;
2007-11-30 00:44:34 +03:00
2008-01-10 06:40:56 +03:00
ctdb - > monitor - > monitoring_mode = CTDB_MONITORING_ACTIVE ;
2008-02-04 12:07:15 +03:00
DEBUG ( DEBUG_NOTICE , ( " Monitoring has been started \n " ) ) ;
2007-05-18 14:06:29 +04:00
}
2007-06-07 05:15:22 +04:00
/*
2007-06-07 09:18:55 +04:00
modify flags on a node
2007-06-07 05:15:22 +04:00
*/
2007-06-07 09:18:55 +04:00
int32_t ctdb_control_modflags ( struct ctdb_context * ctdb , TDB_DATA indata )
2007-06-07 05:15:22 +04:00
{
2008-11-19 06:43:46 +03:00
struct ctdb_node_flag_change * c = ( struct ctdb_node_flag_change * ) indata . dptr ;
struct ctdb_node * node ;
if ( c - > pnn > = ctdb - > num_nodes ) {
DEBUG ( DEBUG_ERR , ( __location__ " Node %d is invalid, num_nodes :%d \n " , c - > pnn , ctdb - > num_nodes ) ) ;
return - 1 ;
}
2007-06-07 05:15:22 +04:00
2008-11-19 06:43:46 +03:00
node = ctdb - > nodes [ c - > pnn ] ;
c - > old_flags = node - > flags ;
node - > flags = c - > new_flags & ~ NODE_FLAGS_DISCONNECTED ;
node - > flags | = ( c - > old_flags & NODE_FLAGS_DISCONNECTED ) ;
2007-06-07 09:18:55 +04:00
2008-11-19 06:43:46 +03:00
if ( node - > flags = = c - > old_flags ) {
DEBUG ( DEBUG_INFO , ( " Control modflags on node %u - Unchanged - flags 0x%x \n " , c - > pnn , node - > flags ) ) ;
2007-06-07 09:18:55 +04:00
return 0 ;
2007-06-07 05:15:22 +04:00
}
2008-11-19 06:43:46 +03:00
DEBUG ( DEBUG_INFO , ( " Control modflags on node %u - flags now 0x%x \n " , c - > pnn , node - > flags ) ) ;
2007-06-07 10:34:33 +04:00
2007-06-07 05:15:22 +04:00
2008-11-19 06:43:46 +03:00
/* tell the recovery daemon something has changed */
ctdb_daemon_send_message ( ctdb , ctdb - > pnn ,
CTDB_SRVID_SET_NODE_FLAGS , indata ) ;
2007-06-07 10:34:33 +04:00
2008-11-19 06:43:46 +03:00
/* if we have become banned, we should go into recovery mode */
if ( ( node - > flags & NODE_FLAGS_BANNED ) & & ! ( c - > old_flags & NODE_FLAGS_BANNED ) ) {
2007-06-07 10:34:33 +04:00
/* make sure we are frozen */
2008-02-04 12:07:15 +03:00
DEBUG ( DEBUG_NOTICE , ( " This node has been banned - forcing freeze and recovery \n " ) ) ;
2007-08-22 04:38:35 +04:00
/* Reset the generation id to 1 to make us ignore any
REQ / REPLY CALL / DMASTER someone sends to us .
We are now banned so we shouldnt service database calls
anymore .
*/
2007-08-22 06:38:31 +04:00
ctdb - > vnn_map - > generation = INVALID_GENERATION ;
2007-08-22 04:38:35 +04:00
2007-06-10 02:46:33 +04:00
ctdb_start_freeze ( ctdb ) ;
ctdb_release_all_ips ( ctdb ) ;
2007-06-07 10:34:33 +04:00
ctdb - > recovery_mode = CTDB_RECOVERY_ACTIVE ;
}
2007-06-07 05:15:22 +04:00
return 0 ;
}
2008-01-10 06:40:56 +03:00
/*
return the monitoring mode
*/
int32_t ctdb_monitoring_mode ( struct ctdb_context * ctdb )
{
if ( ctdb - > monitor = = NULL ) {
return CTDB_MONITORING_DISABLED ;
}
return ctdb - > monitor - > monitoring_mode ;
}
2008-04-02 04:13:30 +04:00