2007-07-04 11:45:46 +04:00
/*
event script handling
Copyright ( C ) Andrew Tridgell 2007
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-07-04 11:45:46 +04:00
( at your option ) any later version .
This program is distributed in the hope that it will be useful ,
but WITHOUT ANY WARRANTY ; without even the implied warranty of
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE . See the
GNU General Public License for more details .
You should have received a copy of the GNU General Public License
2007-07-10 09:29:31 +04:00
along with this program ; if not , see < http : //www.gnu.org/licenses/>.
2007-07-04 11:45:46 +04:00
*/
# include "includes.h"
2009-10-23 06:55:21 +04:00
# include <time.h>
2007-07-04 11:45:46 +04:00
# include "system/filesys.h"
# include "system/wait.h"
2007-08-20 05:10:30 +04:00
# include "system/dir.h"
# include "system/locale.h"
2007-07-04 11:45:46 +04:00
# include "../include/ctdb_private.h"
# include "lib/events/events.h"
2007-08-15 08:44:03 +04:00
# include "../common/rb_tree.h"
2007-08-20 05:10:30 +04:00
static struct {
struct timeval start ;
const char * script_running ;
} child_state ;
2009-11-24 03:46:49 +03:00
static const char * call_names [ ] = {
" startup " ,
" startrecovery " ,
" recovered " ,
" takeip " ,
" releaseip " ,
" stopped " ,
" monitor " ,
" status " ,
" shutdown " ,
2009-11-26 07:49:49 +03:00
" reload "
2009-11-24 03:46:49 +03:00
} ;
2009-10-28 08:11:54 +03:00
static void ctdb_event_script_timeout ( struct event_context * ev , struct timed_event * te , struct timeval t , void * p ) ;
2007-08-20 05:10:30 +04:00
/*
ctdbd sends us a SIGTERM when we should time out the current script
*/
static void sigterm ( int sig )
{
2009-10-23 06:55:21 +04:00
char tbuf [ 100 ] , buf [ 200 ] ;
time_t t ;
2009-10-14 07:14:28 +04:00
DEBUG ( DEBUG_ERR , ( " Timed out running script '%s' after %.1f seconds pid :%d \n " ,
child_state . script_running , timeval_elapsed ( & child_state . start ) , getpid ( ) ) ) ;
2009-10-23 06:55:21 +04:00
t = time ( NULL ) ;
2009-10-14 07:14:28 +04:00
2009-10-23 06:55:21 +04:00
strftime ( tbuf , sizeof ( tbuf ) - 1 , " %Y%m%d%H%M%S " , localtime ( & t ) ) ;
sprintf ( buf , " pstree -p >/tmp/ctdb.event.%s.%d " , tbuf , getpid ( ) ) ;
system ( buf ) ;
DEBUG ( DEBUG_ERR , ( " Logged timedout eventscript : %s \n " , buf ) ) ;
2009-10-14 07:14:28 +04:00
2007-08-20 05:10:30 +04:00
/* all the child processes will be running in the same process group */
kill ( - getpgrp ( ) , SIGKILL ) ;
2009-10-27 05:51:45 +03:00
_exit ( 1 ) ;
2007-08-20 05:10:30 +04:00
}
2007-07-04 11:45:46 +04:00
2008-06-13 06:18:00 +04:00
struct ctdb_event_script_state {
struct ctdb_context * ctdb ;
pid_t child ;
2009-11-24 03:36:53 +03:00
/* Warning: this can free us! */
2008-06-13 06:18:00 +04:00
void ( * callback ) ( struct ctdb_context * , int , void * ) ;
2009-11-24 03:36:53 +03:00
int cb_status ;
2008-06-13 06:18:00 +04:00
int fd [ 2 ] ;
void * private_data ;
2009-11-24 03:49:58 +03:00
enum ctdb_eventscript_call call ;
2008-06-13 06:18:00 +04:00
const char * options ;
2009-10-28 08:11:54 +03:00
struct timeval timeout ;
2008-06-13 06:18:00 +04:00
} ;
2009-03-23 11:07:45 +03:00
struct ctdb_monitor_script_status {
struct ctdb_monitor_script_status * next ;
const char * name ;
struct timeval start ;
struct timeval finished ;
2009-08-13 07:04:08 +04:00
int32_t disabled ;
2009-03-23 11:07:45 +03:00
int32_t status ;
char * output ;
} ;
2009-11-23 23:40:51 +03:00
struct ctdb_monitor_script_status_ctx {
2009-03-23 11:07:45 +03:00
struct ctdb_monitor_script_status * scripts ;
} ;
/* called from ctdb_logging when we have received output on STDERR from
* one of the eventscripts
*/
int ctdb_log_event_script_output ( struct ctdb_context * ctdb , char * str , uint16_t len )
{
struct ctdb_monitor_script_status * script ;
2009-11-23 23:40:51 +03:00
if ( ctdb - > current_monitor_status_ctx = = NULL ) {
2009-03-23 11:07:45 +03:00
return - 1 ;
}
2009-11-23 23:40:51 +03:00
script = ctdb - > current_monitor_status_ctx - > scripts ;
2009-03-23 11:07:45 +03:00
if ( script = = NULL ) {
return - 1 ;
}
if ( script - > output = = NULL ) {
script - > output = talloc_asprintf ( script , " %*.*s " , len , len , str ) ;
} else {
script - > output = talloc_asprintf_append ( script - > output , " %*.*s " , len , len , str ) ;
}
return 0 ;
}
/* called from the event script child process when we are starting a new
* monitor event
*/
int32_t ctdb_control_event_script_init ( struct ctdb_context * ctdb )
{
DEBUG ( DEBUG_INFO , ( " event script init called \n " ) ) ;
2009-11-23 23:40:51 +03:00
if ( ctdb - > current_monitor_status_ctx = = NULL ) {
DEBUG ( DEBUG_ERR , ( __location__ " current_monitor_status_ctx is NULL when initing script \n " ) ) ;
return - 1 ;
2009-03-23 11:07:45 +03:00
}
return 0 ;
}
/* called from the event script child process when we are star running
* an eventscript
*/
int32_t ctdb_control_event_script_start ( struct ctdb_context * ctdb , TDB_DATA indata )
{
const char * name = ( const char * ) indata . dptr ;
struct ctdb_monitor_script_status * script ;
DEBUG ( DEBUG_INFO , ( " event script start called : %s \n " , name ) ) ;
2009-11-23 23:40:51 +03:00
if ( ctdb - > current_monitor_status_ctx = = NULL ) {
DEBUG ( DEBUG_ERR , ( __location__ " current_monitor_status_ctx is NULL when starting script \n " ) ) ;
2009-03-23 11:07:45 +03:00
return - 1 ;
}
2009-11-23 23:40:51 +03:00
script = talloc_zero ( ctdb - > current_monitor_status_ctx , struct ctdb_monitor_script_status ) ;
2009-03-23 11:07:45 +03:00
if ( script = = NULL ) {
DEBUG ( DEBUG_ERR , ( __location__ " Failed to talloc ctdb_monitor_script_status for script %s \n " , name ) ) ;
return - 1 ;
}
2009-11-23 23:40:51 +03:00
script - > next = ctdb - > current_monitor_status_ctx - > scripts ;
2009-03-23 11:07:45 +03:00
script - > name = talloc_strdup ( script , name ) ;
2009-05-20 14:08:13 +04:00
CTDB_NO_MEMORY ( ctdb , script - > name ) ;
2009-03-23 11:07:45 +03:00
script - > start = timeval_current ( ) ;
2009-11-23 23:40:51 +03:00
ctdb - > current_monitor_status_ctx - > scripts = script ;
2009-10-28 08:11:54 +03:00
2009-03-23 11:07:45 +03:00
return 0 ;
}
/* called from the event script child process when we have finished running
* an eventscript
*/
int32_t ctdb_control_event_script_stop ( struct ctdb_context * ctdb , TDB_DATA indata )
{
int32_t res = * ( ( int32_t * ) indata . dptr ) ;
struct ctdb_monitor_script_status * script ;
2009-11-23 23:40:51 +03:00
if ( ctdb - > current_monitor_status_ctx = = NULL ) {
DEBUG ( DEBUG_ERR , ( __location__ " current_monitor_status_ctx is NULL when script finished \n " ) ) ;
2009-03-23 11:07:45 +03:00
return - 1 ;
}
2009-11-23 23:40:51 +03:00
script = ctdb - > current_monitor_status_ctx - > scripts ;
2009-03-23 11:07:45 +03:00
if ( script = = NULL ) {
DEBUG ( DEBUG_ERR , ( __location__ " script is NULL when the script had finished \n " ) ) ;
return - 1 ;
}
script - > finished = timeval_current ( ) ;
script - > status = res ;
2009-10-28 01:07:43 +03:00
DEBUG ( DEBUG_INFO , ( " event script stop called for script:%s duration:%.1f status:%d \n " , script - > name , timeval_elapsed ( & script - > start ) , ( int ) res ) ) ;
2009-03-23 11:07:45 +03:00
return 0 ;
}
2009-08-13 07:04:08 +04:00
/* called from the event script child process when we have a disabled script
*/
int32_t ctdb_control_event_script_disabled ( struct ctdb_context * ctdb , TDB_DATA indata )
{
2009-10-09 15:22:11 +04:00
const char * name = ( const char * ) indata . dptr ;
2009-08-13 07:04:08 +04:00
struct ctdb_monitor_script_status * script ;
2009-10-09 15:22:11 +04:00
DEBUG ( DEBUG_INFO , ( " event script disabed called for script %s \n " , name ) ) ;
2009-08-13 07:04:08 +04:00
2009-11-23 23:40:51 +03:00
if ( ctdb - > current_monitor_status_ctx = = NULL ) {
DEBUG ( DEBUG_ERR , ( __location__ " current_monitor_status_ctx is NULL when script finished \n " ) ) ;
2009-08-13 07:04:08 +04:00
return - 1 ;
}
2009-11-23 23:40:51 +03:00
script = ctdb - > current_monitor_status_ctx - > scripts ;
2009-08-13 07:04:08 +04:00
if ( script = = NULL ) {
DEBUG ( DEBUG_ERR , ( __location__ " script is NULL when the script had finished \n " ) ) ;
return - 1 ;
}
script - > finished = timeval_current ( ) ;
2009-10-09 15:22:11 +04:00
script - > status = 0 ;
2009-08-13 07:04:08 +04:00
script - > disabled = 1 ;
return 0 ;
}
2009-03-23 11:07:45 +03:00
static struct ctdb_monitoring_wire * marshall_monitoring_scripts ( TALLOC_CTX * mem_ctx , struct ctdb_monitoring_wire * monitoring_scripts , struct ctdb_monitor_script_status * script )
{
struct ctdb_monitoring_script_wire script_wire ;
size_t size ;
if ( script = = NULL ) {
return monitoring_scripts ;
}
monitoring_scripts = marshall_monitoring_scripts ( mem_ctx , monitoring_scripts , script - > next ) ;
if ( monitoring_scripts = = NULL ) {
return NULL ;
}
bzero ( & script_wire , sizeof ( struct ctdb_monitoring_script_wire ) ) ;
strncpy ( script_wire . name , script - > name , MAX_SCRIPT_NAME ) ;
script_wire . start = script - > start ;
script_wire . finished = script - > finished ;
2009-08-13 07:04:08 +04:00
script_wire . disabled = script - > disabled ;
2009-03-23 11:07:45 +03:00
script_wire . status = script - > status ;
if ( script - > output ! = NULL ) {
strncpy ( script_wire . output , script - > output , MAX_SCRIPT_OUTPUT ) ;
}
size = talloc_get_size ( monitoring_scripts ) ;
monitoring_scripts = talloc_realloc_size ( mem_ctx , monitoring_scripts , size + sizeof ( struct ctdb_monitoring_script_wire ) ) ;
if ( monitoring_scripts = = NULL ) {
DEBUG ( DEBUG_ERR , ( __location__ " Failed to talloc_resize monitoring_scripts blob \n " ) ) ;
return NULL ;
}
memcpy ( & monitoring_scripts - > scripts [ monitoring_scripts - > num_scripts ] , & script_wire , sizeof ( script_wire ) ) ;
monitoring_scripts - > num_scripts + + ;
return monitoring_scripts ;
}
2009-12-07 15:39:40 +03:00
/* called from the event script child process when we have completed a
* monitor event
*/
int32_t ctdb_control_event_script_finished ( struct ctdb_context * ctdb )
2009-03-23 11:07:45 +03:00
{
2009-12-07 15:39:40 +03:00
DEBUG ( DEBUG_INFO , ( " event script finished called \n " ) ) ;
2009-03-23 11:07:45 +03:00
2009-12-07 15:39:40 +03:00
if ( ctdb - > current_monitor_status_ctx = = NULL ) {
DEBUG ( DEBUG_ERR , ( __location__ " script_status is NULL when monitoring event finished \n " ) ) ;
2009-03-23 11:07:45 +03:00
return - 1 ;
}
2009-12-07 15:39:40 +03:00
talloc_free ( ctdb - > last_status ) ;
ctdb - > last_status = talloc_size ( ctdb , offsetof ( struct ctdb_monitoring_wire , scripts ) ) ;
if ( ctdb - > last_status = = NULL ) {
DEBUG ( DEBUG_ERR , ( __location__ " failed to talloc last_status \n " ) ) ;
2009-03-23 11:07:45 +03:00
return - 1 ;
}
2009-12-07 15:39:40 +03:00
ctdb - > last_status - > num_scripts = 0 ;
ctdb - > last_status = marshall_monitoring_scripts ( ctdb , ctdb - > last_status , ctdb - > current_monitor_status_ctx - > scripts ) ;
talloc_free ( ctdb - > current_monitor_status_ctx ) ;
ctdb - > current_monitor_status_ctx = NULL ;
return 0 ;
}
int32_t ctdb_control_get_event_script_status ( struct ctdb_context * ctdb , TDB_DATA * outdata )
{
struct ctdb_monitoring_wire * monitoring_scripts = ctdb - > last_status ;
2009-03-23 11:07:45 +03:00
if ( monitoring_scripts = = NULL ) {
2009-12-07 15:39:40 +03:00
DEBUG ( DEBUG_ERR , ( __location__ " last_monitor_status_ctx is NULL when reading status \n " ) ) ;
2009-03-23 11:07:45 +03:00
return - 1 ;
}
outdata - > dsize = talloc_get_size ( monitoring_scripts ) ;
outdata - > dptr = ( uint8_t * ) monitoring_scripts ;
return 0 ;
}
2009-08-13 07:04:08 +04:00
struct ctdb_script_tree_item {
const char * name ;
int32_t is_enabled ;
} ;
struct ctdb_script_list {
struct ctdb_script_list * next ;
const char * name ;
int32_t is_enabled ;
} ;
static struct ctdb_script_list * ctdb_get_script_list ( struct ctdb_context * ctdb , TALLOC_CTX * mem_ctx )
2007-07-04 11:45:46 +04:00
{
2007-08-15 08:44:03 +04:00
DIR * dir ;
struct dirent * de ;
2009-08-13 07:04:08 +04:00
struct stat st ;
trbt_tree_t * tree ;
struct ctdb_script_list * head , * tail , * new_item ;
TALLOC_CTX * tmp_ctx = talloc_new ( ctdb ) ;
struct ctdb_script_tree_item * tree_item ;
2008-10-16 10:57:50 +04:00
int count ;
2007-08-20 05:10:30 +04:00
2007-08-15 08:44:03 +04:00
/*
the service specific event scripts
*/
2007-09-04 03:50:07 +04:00
if ( stat ( ctdb - > event_script_dir , & st ) ! = 0 & &
2007-07-04 11:45:46 +04:00
errno = = ENOENT ) {
2008-02-04 12:07:15 +03:00
DEBUG ( DEBUG_CRIT , ( " No event script directory found at '%s' \n " , ctdb - > event_script_dir ) ) ;
2007-08-15 08:44:03 +04:00
talloc_free ( tmp_ctx ) ;
2009-08-13 07:04:08 +04:00
return NULL ;
2007-08-15 08:44:03 +04:00
}
/* create a tree to store all the script names in */
tree = trbt_create ( tmp_ctx , 0 ) ;
/* scan all directory entries and insert all valid scripts into the
tree
*/
2007-09-04 03:50:07 +04:00
dir = opendir ( ctdb - > event_script_dir ) ;
2007-08-15 08:44:03 +04:00
if ( dir = = NULL ) {
2008-02-04 12:07:15 +03:00
DEBUG ( DEBUG_CRIT , ( " Failed to open event script directory '%s' \n " , ctdb - > event_script_dir ) ) ;
2007-08-15 08:44:03 +04:00
talloc_free ( tmp_ctx ) ;
2009-08-13 07:04:08 +04:00
return NULL ;
2007-07-04 11:45:46 +04:00
}
2008-10-16 10:57:50 +04:00
count = 0 ;
2007-08-15 08:44:03 +04:00
while ( ( de = readdir ( dir ) ) ! = NULL ) {
int namlen ;
2007-08-20 05:10:30 +04:00
unsigned num ;
2007-08-21 03:22:14 +04:00
char * str ;
2007-08-15 08:44:03 +04:00
namlen = strlen ( de - > d_name ) ;
2007-07-04 11:45:46 +04:00
2007-08-15 08:44:03 +04:00
if ( namlen < 3 ) {
continue ;
}
2007-07-04 11:45:46 +04:00
2007-08-15 08:44:03 +04:00
if ( de - > d_name [ namlen - 1 ] = = ' ~ ' ) {
/* skip files emacs left behind */
continue ;
}
2007-07-04 11:45:46 +04:00
2007-08-15 08:44:03 +04:00
if ( de - > d_name [ 2 ] ! = ' . ' ) {
continue ;
}
2007-08-20 05:10:30 +04:00
if ( sscanf ( de - > d_name , " %02u. " , & num ) ! = 1 ) {
2007-08-15 08:44:03 +04:00
continue ;
}
2007-08-21 03:22:14 +04:00
/* Make sure the event script is executable */
2007-09-04 03:50:07 +04:00
str = talloc_asprintf ( tree , " %s/%s " , ctdb - > event_script_dir , de - > d_name ) ;
2007-08-21 03:22:14 +04:00
if ( stat ( str , & st ) ! = 0 ) {
2008-02-04 12:07:15 +03:00
DEBUG ( DEBUG_ERR , ( " Could not stat event script %s. Ignoring this event script \n " , str ) ) ;
2007-08-21 03:22:14 +04:00
continue ;
}
2009-08-13 07:04:08 +04:00
tree_item = talloc ( tree , struct ctdb_script_tree_item ) ;
if ( tree_item = = NULL ) {
DEBUG ( DEBUG_ERR , ( __location__ " Failed to allocate new tree item \n " ) ) ;
talloc_free ( tmp_ctx ) ;
return NULL ;
}
tree_item - > is_enabled = 1 ;
2007-08-21 03:22:14 +04:00
if ( ! ( st . st_mode & S_IXUSR ) ) {
2009-06-01 09:29:36 +04:00
DEBUG ( DEBUG_INFO , ( " Event script %s is not executable. Ignoring this event script \n " , str ) ) ;
2009-08-13 07:04:08 +04:00
tree_item - > is_enabled = 0 ;
2007-08-21 03:22:14 +04:00
}
2009-08-13 07:04:08 +04:00
tree_item - > name = talloc_strdup ( tree_item , de - > d_name ) ;
if ( tree_item - > name = = NULL ) {
DEBUG ( DEBUG_ERR , ( __location__ " Failed to allocate script name. \n " ) ) ;
talloc_free ( tmp_ctx ) ;
return NULL ;
}
2008-10-16 10:57:50 +04:00
/* store the event script in the tree */
2009-08-13 07:04:08 +04:00
trbt_insert32 ( tree , ( num < < 16 ) | count + + , tree_item ) ;
2007-07-04 11:45:46 +04:00
}
2007-08-15 08:44:03 +04:00
closedir ( dir ) ;
2009-08-13 07:04:08 +04:00
head = NULL ;
tail = NULL ;
/* fetch the scripts from the tree one by one and add them to the linked
list
*/
while ( ( tree_item = trbt_findfirstarray32 ( tree , 1 ) ) ! = NULL ) {
new_item = talloc ( tmp_ctx , struct ctdb_script_list ) ;
if ( new_item = = NULL ) {
DEBUG ( DEBUG_ERR , ( __location__ " Failed to allocate new list item \n " ) ) ;
talloc_free ( tmp_ctx ) ;
return NULL ;
}
new_item - > next = NULL ;
new_item - > name = talloc_steal ( new_item , tree_item - > name ) ;
new_item - > is_enabled = tree_item - > is_enabled ;
if ( head = = NULL ) {
head = new_item ;
tail = new_item ;
} else {
tail - > next = new_item ;
tail = new_item ;
}
talloc_steal ( mem_ctx , new_item ) ;
/* remove this script from the tree */
talloc_free ( tree_item ) ;
}
talloc_free ( tmp_ctx ) ;
return head ;
}
/*
2009-11-24 03:38:39 +03:00
Actually run the event script
2009-08-13 07:04:08 +04:00
this function is called and run in the context of a forked child
which allows it to do blocking calls such as system ( )
*/
2009-11-24 03:49:58 +03:00
static int ctdb_run_event_script ( struct ctdb_context * ctdb ,
2009-11-26 07:49:49 +03:00
bool from_user ,
2009-11-24 03:49:58 +03:00
enum ctdb_eventscript_call call ,
const char * options )
2009-08-13 07:04:08 +04:00
{
char * cmdstr ;
int ret ;
TALLOC_CTX * tmp_ctx = talloc_new ( ctdb ) ;
struct ctdb_script_list * scripts , * current ;
2009-11-26 07:49:49 +03:00
if ( ! from_user & & call = = CTDB_EVENT_MONITOR ) {
2009-08-13 07:04:08 +04:00
/* This is running in the forked child process. At this stage
* we want to switch from being a ctdb daemon into being a
* client and connect to the real local daemon .
*/
if ( switch_from_server_to_client ( ctdb ) ! = 0 ) {
DEBUG ( DEBUG_CRIT , ( __location__ " ERROR: failed to switch eventscript child into client mode. shutting down. \n " ) ) ;
_exit ( 1 ) ;
}
if ( ctdb_ctrl_event_script_init ( ctdb ) ! = 0 ) {
DEBUG ( DEBUG_ERR , ( __location__ " Failed to init event script monitoring \n " ) ) ;
talloc_free ( tmp_ctx ) ;
return - 1 ;
}
}
if ( ctdb - > recovery_mode ! = CTDB_RECOVERY_NORMAL ) {
/* we guarantee that only some specifically allowed event scripts are run
while in recovery */
2009-11-24 03:49:58 +03:00
const enum ctdb_eventscript_call allowed_calls [ ] = {
CTDB_EVENT_START_RECOVERY , CTDB_EVENT_SHUTDOWN , CTDB_EVENT_RELEASE_IP , CTDB_EVENT_STOPPED } ;
2009-08-13 07:04:08 +04:00
int i ;
2009-11-24 03:49:58 +03:00
for ( i = 0 ; i < ARRAY_SIZE ( allowed_calls ) ; i + + ) {
if ( call = = allowed_calls [ i ] ) break ;
2009-08-13 07:04:08 +04:00
}
2009-11-24 03:49:58 +03:00
if ( i = = ARRAY_SIZE ( allowed_calls ) ) {
DEBUG ( DEBUG_ERR , ( " Refusing to run event scripts call '%s' while in recovery \n " ,
call_names [ call ] ) ) ;
2009-08-13 07:04:08 +04:00
talloc_free ( tmp_ctx ) ;
return - 1 ;
}
}
if ( setpgid ( 0 , 0 ) ! = 0 ) {
DEBUG ( DEBUG_ERR , ( " Failed to create process group for event scripts - %s \n " ,
strerror ( errno ) ) ) ;
talloc_free ( tmp_ctx ) ;
return - 1 ;
}
signal ( SIGTERM , sigterm ) ;
child_state . start = timeval_current ( ) ;
child_state . script_running = " startup " ;
scripts = ctdb_get_script_list ( ctdb , tmp_ctx ) ;
2007-08-15 08:44:03 +04:00
/* fetch the scripts from the tree one by one and execute
them
*/
2009-08-13 07:04:08 +04:00
for ( current = scripts ; current ; current = current - > next ) {
2009-11-26 07:49:49 +03:00
const char * str = from_user ? " CTDB_CALLED_BY_USER=1 " : " " ;
2009-11-13 04:37:55 +03:00
/* Allow a setting where we run the actual monitor event
from an external source and replace it with
a " status " event that just picks up the actual
status of the event asynchronously .
*/
if ( ( ctdb - > tunable . use_status_events_for_monitoring ! = 0 )
2009-11-26 07:49:49 +03:00
& & ( call = = CTDB_EVENT_MONITOR )
& & ! from_user ) {
cmdstr = talloc_asprintf ( tmp_ctx , " %s%s/%s %s " ,
str ,
2009-11-13 04:37:55 +03:00
ctdb - > event_script_dir ,
current - > name , " status " ) ;
} else {
2009-11-26 07:49:49 +03:00
cmdstr = talloc_asprintf ( tmp_ctx , " %s%s/%s %s %s " ,
str ,
2009-11-13 04:37:55 +03:00
ctdb - > event_script_dir ,
2009-11-24 03:52:46 +03:00
current - > name , call_names [ call ] , options ) ;
2009-11-13 04:37:55 +03:00
}
2007-08-15 08:44:03 +04:00
CTDB_NO_MEMORY ( ctdb , cmdstr ) ;
2007-07-04 11:45:46 +04:00
2008-02-04 09:44:24 +03:00
DEBUG ( DEBUG_INFO , ( " Executing event script %s \n " , cmdstr ) ) ;
2007-07-04 11:45:46 +04:00
2007-08-20 05:10:30 +04:00
child_state . start = timeval_current ( ) ;
child_state . script_running = cmdstr ;
2009-11-26 07:49:49 +03:00
if ( ! from_user & & call = = CTDB_EVENT_MONITOR ) {
2009-08-13 07:04:08 +04:00
if ( ctdb_ctrl_event_script_start ( ctdb , current - > name ) ! = 0 ) {
2009-03-23 11:07:45 +03:00
DEBUG ( DEBUG_ERR , ( __location__ " Failed to start event script monitoring \n " ) ) ;
talloc_free ( tmp_ctx ) ;
return - 1 ;
}
2009-08-13 07:04:08 +04:00
if ( ! current - > is_enabled ) {
if ( ctdb_ctrl_event_script_disabled ( ctdb , current - > name ) ! = 0 ) {
DEBUG ( DEBUG_ERR , ( __location__ " Failed to report disabled eventscript \n " ) ) ;
talloc_free ( tmp_ctx ) ;
return - 1 ;
}
}
}
if ( ! current - > is_enabled ) {
continue ;
2009-03-23 11:07:45 +03:00
}
2007-08-15 08:44:03 +04:00
ret = system ( cmdstr ) ;
/* if the system() call was successful, translate ret into the
return code from the command
*/
if ( ret ! = - 1 ) {
ret = WEXITSTATUS ( ret ) ;
}
2009-10-21 09:50:39 +04:00
if ( ret = = 127 ) {
ret = 0 ;
DEBUG ( DEBUG_ERR , ( " Script %s returned status 127. Someone just deleted it? \n " , cmdstr ) ) ;
}
2009-11-26 07:49:49 +03:00
if ( ! from_user & & call = = CTDB_EVENT_MONITOR ) {
2009-03-23 11:07:45 +03:00
if ( ctdb_ctrl_event_script_stop ( ctdb , ret ) ! = 0 ) {
DEBUG ( DEBUG_ERR , ( __location__ " Failed to stop event script monitoring \n " ) ) ;
talloc_free ( tmp_ctx ) ;
return - 1 ;
}
}
2007-08-15 08:44:03 +04:00
/* return an error if the script failed */
if ( ret ! = 0 ) {
2008-02-04 12:07:15 +03:00
DEBUG ( DEBUG_ERR , ( " Event script %s failed with error %d \n " , cmdstr , ret ) ) ;
2009-11-26 07:49:49 +03:00
if ( ! from_user & & call = = CTDB_EVENT_MONITOR ) {
2009-03-23 11:07:45 +03:00
if ( ctdb_ctrl_event_script_finished ( ctdb ) ! = 0 ) {
DEBUG ( DEBUG_ERR , ( __location__ " Failed to finish event script monitoring \n " ) ) ;
talloc_free ( tmp_ctx ) ;
return - 1 ;
}
}
2007-08-15 08:44:03 +04:00
talloc_free ( tmp_ctx ) ;
return ret ;
}
}
2007-08-20 05:10:30 +04:00
child_state . start = timeval_current ( ) ;
child_state . script_running = " finished " ;
2007-08-15 08:44:03 +04:00
2009-11-26 07:49:49 +03:00
if ( ! from_user & & call = = CTDB_EVENT_MONITOR ) {
2009-03-23 11:07:45 +03:00
if ( ctdb_ctrl_event_script_finished ( ctdb ) ! = 0 ) {
DEBUG ( DEBUG_ERR , ( __location__ " Failed to finish event script monitoring \n " ) ) ;
talloc_free ( tmp_ctx ) ;
return - 1 ;
}
}
2007-08-15 08:44:03 +04:00
talloc_free ( tmp_ctx ) ;
return 0 ;
2007-07-04 11:45:46 +04:00
}
/* called when child is finished */
static void ctdb_event_script_handler ( struct event_context * ev , struct fd_event * fde ,
uint16_t flags , void * p )
{
struct ctdb_event_script_state * state =
talloc_get_type ( p , struct ctdb_event_script_state ) ;
struct ctdb_context * ctdb = state - > ctdb ;
2008-07-09 08:02:54 +04:00
2009-11-24 03:36:53 +03:00
if ( read ( state - > fd [ 0 ] , & state - > cb_status , sizeof ( state - > cb_status ) ) ! =
sizeof ( state - > cb_status ) ) {
state - > cb_status = - 2 ;
2009-10-28 08:11:54 +03:00
}
2009-11-24 03:52:46 +03:00
DEBUG ( DEBUG_INFO , ( __location__ " Eventscript %s %s finished with state %d \n " ,
call_names [ state - > call ] , state - > options , state - > cb_status ) ) ;
2009-11-23 23:40:51 +03:00
2009-11-24 03:36:53 +03:00
state - > child = 0 ;
2008-06-13 07:18:06 +04:00
ctdb - > event_script_timeouts = 0 ;
2007-07-04 11:45:46 +04:00
talloc_free ( state ) ;
2008-06-13 07:18:06 +04:00
}
static void ctdb_ban_self ( struct ctdb_context * ctdb , uint32_t ban_period )
{
TDB_DATA data ;
2009-09-03 20:20:39 +04:00
struct ctdb_ban_time bantime ;
2008-06-13 07:18:06 +04:00
2009-09-03 20:20:39 +04:00
bantime . pnn = ctdb - > pnn ;
bantime . time = ban_period ;
2008-06-13 07:18:06 +04:00
2009-09-03 20:20:39 +04:00
data . dsize = sizeof ( bantime ) ;
data . dptr = ( uint8_t * ) & bantime ;
2008-06-13 07:18:06 +04:00
2009-09-03 20:20:39 +04:00
ctdb_control_set_ban_state ( ctdb , data ) ;
2007-07-04 11:45:46 +04:00
}
/* called when child times out */
static void ctdb_event_script_timeout ( struct event_context * ev , struct timed_event * te ,
struct timeval t , void * p )
{
struct ctdb_event_script_state * state = talloc_get_type ( p , struct ctdb_event_script_state ) ;
struct ctdb_context * ctdb = state - > ctdb ;
2009-10-28 08:11:54 +03:00
2009-11-24 03:52:46 +03:00
DEBUG ( DEBUG_ERR , ( " Event script timed out : %s %s count : %u pid : %d \n " ,
call_names [ state - > call ] , state - > options , ctdb - > event_script_timeouts , state - > child ) ) ;
2007-07-04 11:45:46 +04:00
2009-10-14 07:14:28 +04:00
if ( kill ( state - > child , 0 ) ! = 0 ) {
DEBUG ( DEBUG_ERR , ( " Event script child process already dead, errno %s(%d) \n " , strerror ( errno ) , errno ) ) ;
2009-11-24 03:36:53 +03:00
state - > child = 0 ;
2009-10-14 07:14:28 +04:00
talloc_free ( state ) ;
return ;
}
2008-06-13 07:18:06 +04:00
2009-11-24 03:49:58 +03:00
if ( state - > call = = CTDB_EVENT_MONITOR ) {
2008-07-07 14:38:59 +04:00
/* if it is a monitor event, we allow it to "hang" a few times
before we declare it a failure and ban ourself ( and make
ourself unhealthy )
*/
DEBUG ( DEBUG_ERR , ( __location__ " eventscript for monitor event timedout. \n " ) ) ;
ctdb - > event_script_timeouts + + ;
2009-11-23 23:40:51 +03:00
2008-07-07 14:38:59 +04:00
if ( ctdb - > event_script_timeouts > ctdb - > tunable . script_ban_count ) {
2009-11-23 23:40:51 +03:00
DEBUG ( DEBUG_ERR , ( " Maximum timeout count %u reached for eventscript. Making node unhealthy \n " , ctdb - > tunable . script_ban_count ) ) ;
2009-11-24 03:36:53 +03:00
state - > cb_status = - ETIME ;
2008-07-07 14:38:59 +04:00
} else {
2009-11-24 03:36:53 +03:00
state - > cb_status = 0 ;
2008-07-07 14:38:59 +04:00
}
2009-11-24 03:49:58 +03:00
} else if ( state - > call = = CTDB_EVENT_STARTUP ) {
2008-07-07 14:38:59 +04:00
DEBUG ( DEBUG_ERR , ( __location__ " eventscript for startup event timedout. \n " ) ) ;
2009-12-07 15:39:42 +03:00
state - > cb_status = - ETIME ;
2008-07-07 14:38:59 +04:00
} else {
2009-11-23 23:40:51 +03:00
/* if it is not a monitor or a startup event we ban ourself
immediately
*/
2008-07-07 14:38:59 +04:00
DEBUG ( DEBUG_ERR , ( __location__ " eventscript for NON-monitor/NON-startup event timedout. Immediately banning ourself for %d seconds \n " , ctdb - > tunable . recovery_ban_period ) ) ;
2009-11-23 23:40:51 +03:00
2008-06-13 07:18:06 +04:00
ctdb_ban_self ( ctdb , ctdb - > tunable . recovery_ban_period ) ;
2009-11-23 23:40:51 +03:00
2009-12-07 15:39:42 +03:00
state - > cb_status = - ETIME ;
2008-06-13 07:18:06 +04:00
}
2008-07-07 14:38:59 +04:00
2009-11-24 03:49:58 +03:00
if ( state - > call = = CTDB_EVENT_MONITOR | | state - > call = = CTDB_EVENT_STATUS ) {
2009-03-23 11:07:45 +03:00
struct ctdb_monitor_script_status * script ;
2009-11-23 23:40:51 +03:00
if ( ctdb - > current_monitor_status_ctx = = NULL ) {
talloc_free ( state ) ;
return ;
}
script = ctdb - > current_monitor_status_ctx - > scripts ;
2009-03-23 11:07:45 +03:00
if ( script ! = NULL ) {
2009-12-07 15:39:42 +03:00
script - > status = state - > cb_status ;
2009-03-23 11:07:45 +03:00
}
2009-11-23 23:40:51 +03:00
2009-12-02 08:45:57 +03:00
ctdb_control_event_script_finished ( ctdb ) ;
2009-03-23 11:07:45 +03:00
}
2009-11-03 22:01:00 +03:00
talloc_free ( state ) ;
2007-07-04 11:45:46 +04:00
}
/*
2009-11-24 03:36:53 +03:00
destroy an event script : kill it if - > child ! = 0.
2007-07-04 11:45:46 +04:00
*/
static int event_script_destructor ( struct ctdb_event_script_state * state )
{
2009-11-24 03:36:53 +03:00
if ( state - > child ) {
DEBUG ( DEBUG_ERR , ( __location__ " Sending SIGTERM to child pid:%d \n " , state - > child ) ) ;
2009-10-14 07:14:28 +04:00
2009-11-24 03:36:53 +03:00
if ( kill ( state - > child , SIGTERM ) ! = 0 ) {
DEBUG ( DEBUG_ERR , ( " Failed to kill child process for eventscript, errno %s(%d) \n " , strerror ( errno ) , errno ) ) ;
}
2009-10-28 08:11:54 +03:00
}
2009-11-24 03:36:53 +03:00
/* This is allowed to free us; talloc will prevent double free anyway,
* but beware if you call this outside the destructor ! */
if ( state - > callback ) {
state - > callback ( state - > ctdb , state - > cb_status , state - > private_data ) ;
2009-10-14 07:14:28 +04:00
}
2007-07-04 11:45:46 +04:00
return 0 ;
}
2009-11-24 03:53:13 +03:00
static unsigned int count_words ( const char * options )
{
unsigned int words = 0 ;
options + = strspn ( options , " \t " ) ;
while ( * options ) {
words + + ;
options + = strcspn ( options , " \t " ) ;
options + = strspn ( options , " \t " ) ;
}
return words ;
}
static bool check_options ( enum ctdb_eventscript_call call , const char * options )
{
switch ( call ) {
/* These all take no arguments. */
case CTDB_EVENT_STARTUP :
case CTDB_EVENT_START_RECOVERY :
case CTDB_EVENT_RECOVERED :
case CTDB_EVENT_STOPPED :
case CTDB_EVENT_MONITOR :
case CTDB_EVENT_STATUS :
case CTDB_EVENT_SHUTDOWN :
2009-11-27 04:04:02 +03:00
case CTDB_EVENT_RELOAD :
2009-11-24 03:53:13 +03:00
return count_words ( options ) = = 0 ;
case CTDB_EVENT_TAKE_IP : /* interface, IP address, netmask bits. */
case CTDB_EVENT_RELEASE_IP :
return count_words ( options ) = = 3 ;
default :
DEBUG ( DEBUG_ERR , ( __location__ " Unknown ctdb_eventscript_call %u \n " , call ) ) ;
return false ;
}
}
2007-07-04 11:45:46 +04:00
/*
run the event script in the background , calling the callback when
finished
*/
2007-07-19 07:36:00 +04:00
static int ctdb_event_script_callback_v ( struct ctdb_context * ctdb ,
void ( * callback ) ( struct ctdb_context * , int , void * ) ,
void * private_data ,
2009-11-26 07:49:49 +03:00
bool from_user ,
2009-11-24 03:46:49 +03:00
enum ctdb_eventscript_call call ,
2007-07-19 07:36:00 +04:00
const char * fmt , va_list ap )
2007-07-04 11:45:46 +04:00
{
2009-11-23 23:40:51 +03:00
TALLOC_CTX * mem_ctx ;
2007-07-04 11:45:46 +04:00
struct ctdb_event_script_state * state ;
int ret ;
2009-11-26 07:49:49 +03:00
if ( ! from_user & & ( call = = CTDB_EVENT_MONITOR | | call = = CTDB_EVENT_STATUS ) ) {
2009-11-23 23:40:51 +03:00
/* if this was a "monitor" or a status event, we recycle the
context to start a new monitor event
*/
if ( ctdb - > monitor_event_script_ctx ! = NULL ) {
talloc_free ( ctdb - > monitor_event_script_ctx ) ;
ctdb - > monitor_event_script_ctx = NULL ;
2009-10-28 09:35:15 +03:00
}
2009-11-23 23:40:51 +03:00
ctdb - > monitor_event_script_ctx = talloc_new ( ctdb ) ;
mem_ctx = ctdb - > monitor_event_script_ctx ;
if ( ctdb - > current_monitor_status_ctx ! = NULL ) {
talloc_free ( ctdb - > current_monitor_status_ctx ) ;
ctdb - > current_monitor_status_ctx = NULL ;
2009-10-28 09:35:15 +03:00
}
2009-11-25 03:30:11 +03:00
ctdb - > current_monitor_status_ctx = talloc ( ctdb , struct ctdb_monitor_script_status_ctx ) ;
CTDB_NO_MEMORY ( ctdb , ctdb - > current_monitor_status_ctx ) ;
2009-11-23 23:40:51 +03:00
ctdb - > current_monitor_status_ctx - > scripts = NULL ;
} else {
/* any other script will first terminate any monitor event */
if ( ctdb - > monitor_event_script_ctx ! = NULL ) {
talloc_free ( ctdb - > monitor_event_script_ctx ) ;
ctdb - > monitor_event_script_ctx = NULL ;
}
/* and then use a context common for all non-monitor events */
2009-11-25 03:30:11 +03:00
if ( ctdb - > other_event_script_ctx = = NULL ) {
ctdb - > other_event_script_ctx = talloc_new ( ctdb ) ;
}
2009-11-23 23:40:51 +03:00
mem_ctx = ctdb - > other_event_script_ctx ;
2009-10-28 08:11:54 +03:00
}
2009-11-23 23:40:51 +03:00
state = talloc ( mem_ctx , struct ctdb_event_script_state ) ;
CTDB_NO_MEMORY ( ctdb , state ) ;
2007-07-04 11:45:46 +04:00
state - > ctdb = ctdb ;
state - > callback = callback ;
state - > private_data = private_data ;
2009-11-24 03:49:58 +03:00
state - > call = call ;
2008-06-13 06:18:00 +04:00
state - > options = talloc_vasprintf ( state , fmt , ap ) ;
2009-11-24 03:39:46 +03:00
state - > timeout = timeval_set ( ctdb - > tunable . script_timeout , 0 ) ;
2009-10-28 08:11:54 +03:00
if ( state - > options = = NULL ) {
DEBUG ( DEBUG_ERR , ( __location__ " could not allocate state->options \n " ) ) ;
2009-10-28 09:35:15 +03:00
talloc_free ( state ) ;
2009-10-28 08:11:54 +03:00
return - 1 ;
}
2009-11-24 03:53:13 +03:00
if ( ! check_options ( state - > call , state - > options ) ) {
DEBUG ( DEBUG_ERR , ( " Bad eventscript options '%s' for %s \n " ,
call_names [ state - > call ] , state - > options ) ) ;
talloc_free ( state ) ;
return - 1 ;
}
2009-09-09 04:57:39 +04:00
2009-11-24 03:52:46 +03:00
DEBUG ( DEBUG_INFO , ( __location__ " Starting eventscript %s %s \n " ,
call_names [ state - > call ] , state - > options ) ) ;
2007-07-04 11:45:46 +04:00
ret = pipe ( state - > fd ) ;
if ( ret ! = 0 ) {
2009-10-28 09:35:15 +03:00
talloc_free ( state ) ;
2007-07-04 11:45:46 +04:00
return - 1 ;
}
state - > child = fork ( ) ;
if ( state - > child = = ( pid_t ) - 1 ) {
close ( state - > fd [ 0 ] ) ;
close ( state - > fd [ 1 ] ) ;
2009-10-28 09:35:15 +03:00
talloc_free ( state ) ;
2007-07-04 11:45:46 +04:00
return - 1 ;
}
if ( state - > child = = 0 ) {
2009-11-24 03:30:13 +03:00
int rt ;
2008-07-09 08:02:54 +04:00
2007-07-04 11:45:46 +04:00
close ( state - > fd [ 0 ] ) ;
set_close_on_exec ( state - > fd [ 1 ] ) ;
2009-03-23 11:07:45 +03:00
2009-11-26 07:49:49 +03:00
rt = ctdb_run_event_script ( ctdb , from_user , state - > call , state - > options ) ;
2009-11-24 03:30:13 +03:00
/* We must be able to write PIPEBUF bytes at least; if this
somehow fails , the read above will be short . */
2009-11-23 23:40:51 +03:00
write ( state - > fd [ 1 ] , & rt , sizeof ( rt ) ) ;
close ( state - > fd [ 1 ] ) ;
2008-07-09 08:02:54 +04:00
_exit ( rt ) ;
2007-07-04 11:45:46 +04:00
}
close ( state - > fd [ 1 ] ) ;
2009-10-02 07:41:54 +04:00
set_close_on_exec ( state - > fd [ 0 ] ) ;
2009-11-23 23:40:51 +03:00
talloc_set_destructor ( state , event_script_destructor ) ;
2007-07-04 11:45:46 +04:00
2009-10-21 08:26:24 +04:00
DEBUG ( DEBUG_DEBUG , ( __location__ " Created PIPE FD:%d to child eventscript process \n " , state - > fd [ 0 ] ) ) ;
2009-10-15 04:24:54 +04:00
2007-07-04 11:45:46 +04:00
event_add_fd ( ctdb - > ev , state , state - > fd [ 0 ] , EVENT_FD_READ | EVENT_FD_AUTOCLOSE ,
ctdb_event_script_handler , state ) ;
2009-10-28 08:11:54 +03:00
if ( ! timeval_is_zero ( & state - > timeout ) ) {
2009-11-23 23:40:51 +03:00
event_add_timed ( ctdb - > ev , state , timeval_current_ofs ( state - > timeout . tv_sec , state - > timeout . tv_usec ) , ctdb_event_script_timeout , state ) ;
2008-06-13 06:18:00 +04:00
} else {
2009-11-24 03:52:46 +03:00
DEBUG ( DEBUG_ERR , ( __location__ " eventscript %s %s called with no timeout \n " ,
call_names [ state - > call ] , state - > options ) ) ;
2007-07-04 11:45:46 +04:00
}
return 0 ;
}
2007-07-19 07:36:00 +04:00
/*
run the event script in the background , calling the callback when
finished
*/
int ctdb_event_script_callback ( struct ctdb_context * ctdb ,
TALLOC_CTX * mem_ctx ,
void ( * callback ) ( struct ctdb_context * , int , void * ) ,
void * private_data ,
2009-11-26 07:49:49 +03:00
bool from_user ,
2009-11-24 03:46:49 +03:00
enum ctdb_eventscript_call call ,
2007-07-19 07:36:00 +04:00
const char * fmt , . . . )
{
va_list ap ;
int ret ;
va_start ( ap , fmt ) ;
2009-11-26 07:49:49 +03:00
ret = ctdb_event_script_callback_v ( ctdb , callback , private_data , from_user , call , fmt , ap ) ;
2007-07-19 07:36:00 +04:00
va_end ( ap ) ;
return ret ;
}
struct callback_status {
bool done ;
int status ;
} ;
/*
called when ctdb_event_script ( ) finishes
*/
static void event_script_callback ( struct ctdb_context * ctdb , int status , void * private_data )
{
struct callback_status * s = ( struct callback_status * ) private_data ;
s - > done = true ;
s - > status = status ;
}
/*
2009-11-23 23:40:51 +03:00
run the event script , waiting for it to complete . Used when the caller
doesn ' t want to continue till the event script has finished .
2007-07-19 07:36:00 +04:00
*/
2009-11-24 03:46:49 +03:00
int ctdb_event_script_args ( struct ctdb_context * ctdb , enum ctdb_eventscript_call call ,
const char * fmt , . . . )
2007-07-19 07:36:00 +04:00
{
va_list ap ;
int ret ;
struct callback_status status ;
va_start ( ap , fmt ) ;
2009-11-24 03:46:49 +03:00
ret = ctdb_event_script_callback_v ( ctdb ,
2009-11-26 07:49:49 +03:00
event_script_callback , & status , false , call , fmt , ap ) ;
2007-07-19 07:36:00 +04:00
if ( ret ! = 0 ) {
return ret ;
}
2009-11-24 03:46:49 +03:00
va_end ( ap ) ;
2007-07-19 07:36:00 +04:00
status . status = - 1 ;
status . done = false ;
while ( status . done = = false & & event_loop_once ( ctdb - > ev ) = = 0 ) /* noop */ ;
return status . status ;
}
2008-04-02 04:13:30 +04:00
2009-11-24 03:46:49 +03:00
int ctdb_event_script ( struct ctdb_context * ctdb , enum ctdb_eventscript_call call )
{
/* GCC complains about empty format string, so use %s and "". */
return ctdb_event_script_args ( ctdb , call , " %s " , " " ) ;
}
2008-04-02 04:13:30 +04:00
struct eventscript_callback_state {
struct ctdb_req_control * c ;
} ;
/*
2009-11-19 03:03:51 +03:00
called when a forced eventscript run has finished
2008-04-02 04:13:30 +04:00
*/
static void run_eventscripts_callback ( struct ctdb_context * ctdb , int status ,
void * private_data )
{
struct eventscript_callback_state * state =
talloc_get_type ( private_data , struct eventscript_callback_state ) ;
ctdb_enable_monitoring ( ctdb ) ;
if ( status ! = 0 ) {
DEBUG ( DEBUG_ERR , ( __location__ " Failed to forcibly run eventscripts \n " ) ) ;
}
2009-11-23 23:40:51 +03:00
ctdb_request_control_reply ( ctdb , state - > c , NULL , status , NULL ) ;
2009-11-24 03:36:53 +03:00
/* This will free the struct ctdb_event_script_state we are in! */
2008-04-02 04:13:30 +04:00
talloc_free ( state ) ;
return ;
}
2009-11-23 23:40:51 +03:00
2009-11-25 03:32:29 +03:00
/* Returns rest of string, or NULL if no match. */
2009-11-24 03:54:22 +03:00
static const char * get_call ( const char * p , enum ctdb_eventscript_call * call )
{
unsigned int len ;
/* Skip any initial whitespace. */
p + = strspn ( p , " \t " ) ;
/* See if we match any. */
for ( * call = 0 ; * call < ARRAY_SIZE ( call_names ) ; ( * call ) + + ) {
len = strlen ( call_names [ * call ] ) ;
if ( strncmp ( p , call_names [ * call ] , len ) = = 0 ) {
2009-11-25 03:32:29 +03:00
/* If end of string or whitespace, we're done. */
if ( strcspn ( p + len , " \t " ) = = 0 ) {
return p + len ;
}
2009-11-24 03:54:22 +03:00
}
}
return NULL ;
}
2008-04-02 04:13:30 +04:00
/*
A control to force running of the eventscripts from the ctdb client tool
*/
int32_t ctdb_run_eventscripts ( struct ctdb_context * ctdb ,
struct ctdb_req_control * c ,
TDB_DATA indata , bool * async_reply )
{
int ret ;
struct eventscript_callback_state * state ;
2009-11-24 03:54:22 +03:00
const char * options ;
enum ctdb_eventscript_call call ;
/* Figure out what call they want. */
options = get_call ( ( const char * ) indata . dptr , & call ) ;
if ( ! options ) {
DEBUG ( DEBUG_ERR , ( __location__ " Invalid forced \" %s \" \n " , ( const char * ) indata . dptr ) ) ;
return - 1 ;
}
2008-04-02 04:13:30 +04:00
2009-11-23 23:40:51 +03:00
if ( ctdb - > recovery_mode ! = CTDB_RECOVERY_NORMAL ) {
DEBUG ( DEBUG_ERR , ( __location__ " Aborted running eventscript \" %s \" while in RECOVERY mode \n " , indata . dptr ) ) ;
return - 1 ;
2008-04-02 04:13:30 +04:00
}
2009-11-23 23:40:51 +03:00
state = talloc ( ctdb - > other_event_script_ctx , struct eventscript_callback_state ) ;
2008-04-02 04:13:30 +04:00
CTDB_NO_MEMORY ( ctdb , state ) ;
2008-07-11 04:33:46 +04:00
state - > c = talloc_steal ( state , c ) ;
2008-04-02 04:13:30 +04:00
DEBUG ( DEBUG_NOTICE , ( " Forced running of eventscripts with arguments %s \n " , indata . dptr ) ) ;
ctdb_disable_monitoring ( ctdb ) ;
ret = ctdb_event_script_callback ( ctdb ,
state , run_eventscripts_callback , state ,
2009-11-26 07:49:49 +03:00
true , call , " %s " , options ) ;
2008-04-02 04:13:30 +04:00
if ( ret ! = 0 ) {
ctdb_enable_monitoring ( ctdb ) ;
DEBUG ( DEBUG_ERR , ( __location__ " Failed to run eventscripts with arguments %s \n " , indata . dptr ) ) ;
talloc_free ( state ) ;
return - 1 ;
}
/* tell ctdb_control.c that we will be replying asynchronously */
* async_reply = true ;
return 0 ;
}
2009-08-13 07:04:08 +04:00
int32_t ctdb_control_enable_script ( struct ctdb_context * ctdb , TDB_DATA indata )
{
const char * script ;
struct stat st ;
char * filename ;
TALLOC_CTX * tmp_ctx = talloc_new ( ctdb ) ;
script = ( char * ) indata . dptr ;
if ( indata . dsize = = 0 ) {
DEBUG ( DEBUG_ERR , ( __location__ " No script specified. \n " ) ) ;
talloc_free ( tmp_ctx ) ;
return - 1 ;
}
if ( indata . dptr [ indata . dsize - 1 ] ! = ' \0 ' ) {
DEBUG ( DEBUG_ERR , ( __location__ " String is not null terminated. \n " ) ) ;
talloc_free ( tmp_ctx ) ;
return - 1 ;
}
if ( index ( script , ' / ' ) ! = NULL ) {
DEBUG ( DEBUG_ERR , ( __location__ " Script name contains '/'. Failed to enable script %s \n " , script ) ) ;
talloc_free ( tmp_ctx ) ;
return - 1 ;
}
if ( stat ( ctdb - > event_script_dir , & st ) ! = 0 & &
errno = = ENOENT ) {
DEBUG ( DEBUG_CRIT , ( " No event script directory found at '%s' \n " , ctdb - > event_script_dir ) ) ;
talloc_free ( tmp_ctx ) ;
return - 1 ;
}
filename = talloc_asprintf ( tmp_ctx , " %s/%s " , ctdb - > event_script_dir , script ) ;
if ( filename = = NULL ) {
DEBUG ( DEBUG_ERR , ( __location__ " Failed to create script path \n " ) ) ;
talloc_free ( tmp_ctx ) ;
return - 1 ;
}
if ( stat ( filename , & st ) ! = 0 ) {
DEBUG ( DEBUG_ERR , ( " Could not stat event script %s. Failed to enable script. \n " , filename ) ) ;
talloc_free ( tmp_ctx ) ;
return - 1 ;
}
if ( chmod ( filename , st . st_mode | S_IXUSR ) = = - 1 ) {
DEBUG ( DEBUG_ERR , ( " Could not chmod %s. Failed to enable script. \n " , filename ) ) ;
talloc_free ( tmp_ctx ) ;
return - 1 ;
}
talloc_free ( tmp_ctx ) ;
return 0 ;
}
int32_t ctdb_control_disable_script ( struct ctdb_context * ctdb , TDB_DATA indata )
{
const char * script ;
struct stat st ;
char * filename ;
TALLOC_CTX * tmp_ctx = talloc_new ( ctdb ) ;
script = ( char * ) indata . dptr ;
if ( indata . dsize = = 0 ) {
DEBUG ( DEBUG_ERR , ( __location__ " No script specified. \n " ) ) ;
talloc_free ( tmp_ctx ) ;
return - 1 ;
}
if ( indata . dptr [ indata . dsize - 1 ] ! = ' \0 ' ) {
DEBUG ( DEBUG_ERR , ( __location__ " String is not null terminated. \n " ) ) ;
talloc_free ( tmp_ctx ) ;
return - 1 ;
}
if ( index ( script , ' / ' ) ! = NULL ) {
DEBUG ( DEBUG_ERR , ( __location__ " Script name contains '/'. Failed to disable script %s \n " , script ) ) ;
talloc_free ( tmp_ctx ) ;
return - 1 ;
}
if ( stat ( ctdb - > event_script_dir , & st ) ! = 0 & &
errno = = ENOENT ) {
DEBUG ( DEBUG_CRIT , ( " No event script directory found at '%s' \n " , ctdb - > event_script_dir ) ) ;
talloc_free ( tmp_ctx ) ;
return - 1 ;
}
filename = talloc_asprintf ( tmp_ctx , " %s/%s " , ctdb - > event_script_dir , script ) ;
if ( filename = = NULL ) {
DEBUG ( DEBUG_ERR , ( __location__ " Failed to create script path \n " ) ) ;
talloc_free ( tmp_ctx ) ;
return - 1 ;
}
if ( stat ( filename , & st ) ! = 0 ) {
DEBUG ( DEBUG_ERR , ( " Could not stat event script %s. Failed to disable script. \n " , filename ) ) ;
talloc_free ( tmp_ctx ) ;
return - 1 ;
}
if ( chmod ( filename , st . st_mode & ~ ( S_IXUSR | S_IXGRP | S_IXOTH ) ) = = - 1 ) {
DEBUG ( DEBUG_ERR , ( " Could not chmod %s. Failed to disable script. \n " , filename ) ) ;
talloc_free ( tmp_ctx ) ;
return - 1 ;
}
talloc_free ( tmp_ctx ) ;
return 0 ;
}