2008-01-16 22:03:01 +11:00
/*
ctdb logging code
Copyright ( C ) Andrew Tridgell 2008
This program is free software ; you can redistribute it and / or modify
it under the terms of the GNU General Public License as published by
the Free Software Foundation ; either version 3 of the License , or
( at your option ) any later version .
This program is distributed in the hope that it will be useful ,
but WITHOUT ANY WARRANTY ; without even the implied warranty of
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE . See the
GNU General Public License for more details .
You should have received a copy of the GNU General Public License
along with this program ; if not , see < http : //www.gnu.org/licenses/>.
*/
# include "includes.h"
# include "lib/events/events.h"
2009-10-30 18:53:17 +11:00
# include "../include/ctdb.h"
2008-01-16 22:03:01 +11:00
# include "../include/ctdb_private.h"
# include "system/syslog.h"
# include "system/time.h"
# include "system/filesys.h"
2009-10-30 18:53:17 +11:00
struct syslog_message {
uint32_t level ;
uint32_t len ;
char message [ 1 ] ;
} ;
2009-11-16 15:17:32 +11:00
struct ctdb_syslog_state {
int syslog_fd ;
int fd [ 2 ] ;
} ;
static int syslogd_is_started = 0 ;
/* called when child is finished
* this is for the syslog daemon , we can not use DEBUG here
*/
static void ctdb_syslog_handler ( struct event_context * ev , struct fd_event * fde ,
uint16_t flags , void * p )
{
struct ctdb_syslog_state * state = talloc_get_type ( p , struct ctdb_syslog_state ) ;
int count ;
char str [ 65536 ] ;
struct syslog_message * msg ;
if ( state = = NULL ) {
return ;
}
count = recv ( state - > syslog_fd , str , sizeof ( str ) , 0 ) ;
if ( count < sizeof ( struct syslog_message ) ) {
return ;
}
msg = ( struct syslog_message * ) str ;
syslog ( msg - > level , " %s " , msg - > message ) ;
}
/* called when the pipd from the main daemon has closed
* this is for the syslog daemon , we can not use DEBUG here
*/
static void ctdb_syslog_terminate_handler ( struct event_context * ev , struct fd_event * fde ,
uint16_t flags , void * p )
{
syslog ( LOG_ERR , " Shutting down SYSLOG daemon with pid:%d " , ( int ) getpid ( ) ) ;
_exit ( 0 ) ;
}
2009-10-30 18:53:17 +11:00
/*
* this is for the syslog daemon , we can not use DEBUG here
*/
int start_syslog_daemon ( struct ctdb_context * ctdb )
{
struct sockaddr_in syslog_sin ;
2009-11-16 15:17:32 +11:00
struct ctdb_syslog_state * state ;
2009-10-30 18:53:17 +11:00
2009-11-16 15:17:32 +11:00
state = talloc ( ctdb , struct ctdb_syslog_state ) ;
CTDB_NO_MEMORY ( ctdb , state ) ;
if ( pipe ( state - > fd ) ! = 0 ) {
printf ( " Failed to create syslog pipe \n " ) ;
talloc_free ( state ) ;
return - 1 ;
}
ctdb - > syslogd_pid = fork ( ) ;
if ( ctdb - > syslogd_pid = = ( pid_t ) - 1 ) {
2009-10-30 18:53:17 +11:00
printf ( " Failed to create syslog child process \n " ) ;
2009-11-16 15:17:32 +11:00
close ( state - > fd [ 0 ] ) ;
close ( state - > fd [ 1 ] ) ;
talloc_free ( state ) ;
2009-10-30 18:53:17 +11:00
return - 1 ;
}
2009-10-30 19:39:11 +11:00
syslogd_is_started = 1 ;
2009-11-16 15:17:32 +11:00
if ( ctdb - > syslogd_pid ! = 0 ) {
DEBUG ( DEBUG_ERR , ( " Starting SYSLOG child process with pid:%d \n " , ( int ) ctdb - > syslogd_pid ) ) ;
close ( state - > fd [ 1 ] ) ;
set_close_on_exec ( state - > fd [ 0 ] ) ;
2009-10-30 18:53:17 +11:00
return 0 ;
}
2009-11-17 12:07:10 +11:00
talloc_free ( ctdb - > ev ) ;
ctdb - > ev = event_context_init ( NULL ) ;
2009-11-16 15:17:32 +11:00
syslog ( LOG_ERR , " Starting SYSLOG daemon with pid:%d " , ( int ) getpid ( ) ) ;
close ( state - > fd [ 0 ] ) ;
event_add_fd ( ctdb - > ev , state , state - > fd [ 1 ] , EVENT_FD_READ | EVENT_FD_AUTOCLOSE ,
ctdb_syslog_terminate_handler , state ) ;
state - > syslog_fd = socket ( AF_INET , SOCK_DGRAM , IPPROTO_UDP ) ;
if ( state - > syslog_fd = = - 1 ) {
2009-10-30 18:53:17 +11:00
printf ( " Failed to create syslog socket \n " ) ;
return - 1 ;
}
syslog_sin . sin_family = AF_INET ;
syslog_sin . sin_port = htons ( CTDB_PORT ) ;
syslog_sin . sin_addr . s_addr = htonl ( INADDR_LOOPBACK ) ;
2009-11-16 15:17:32 +11:00
if ( bind ( state - > syslog_fd , & syslog_sin , sizeof ( syslog_sin ) ) = = - 1 ) {
2009-10-30 18:53:17 +11:00
if ( errno = = EADDRINUSE ) {
/* this is ok, we already have a syslog daemon */
_exit ( 0 ) ;
}
printf ( " syslog daemon failed to bind to socket. errno:%d(%s) \n " , errno , strerror ( errno ) ) ;
_exit ( 10 ) ;
}
2009-11-16 15:17:32 +11:00
event_add_fd ( ctdb - > ev , state , state - > syslog_fd , EVENT_FD_READ | EVENT_FD_AUTOCLOSE ,
ctdb_syslog_handler , state ) ;
2009-10-30 18:53:17 +11:00
2009-11-16 15:17:32 +11:00
event_loop_wait ( ctdb - > ev ) ;
/* this should not happen */
2009-10-30 18:53:17 +11:00
_exit ( 10 ) ;
}
2008-01-16 22:03:01 +11:00
struct ctdb_log_state {
2009-12-08 00:31:29 +10:30
struct ctdb_context * ctdb ;
2008-01-16 22:03:01 +11:00
int fd , pfd ;
char buf [ 1024 ] ;
uint16_t buf_used ;
bool use_syslog ;
2009-12-08 12:44:30 +10:30
void ( * logfn ) ( const char * , uint16_t , void * ) ;
void * logfn_private ;
2008-01-16 22:03:01 +11:00
} ;
2009-09-15 19:33:35 +10:00
/* we need this global to keep the DEBUG() syntax */
2008-01-16 22:03:01 +11:00
static struct ctdb_log_state * log_state ;
/*
syslog logging function
*/
static void ctdb_syslog_log ( const char * format , va_list ap )
{
2009-10-30 18:53:17 +11:00
struct syslog_message * msg ;
2008-02-04 17:44:24 +11:00
int level = LOG_DEBUG ;
2009-10-30 18:53:17 +11:00
char * s = NULL ;
int len , ret ;
int syslog_fd ;
struct sockaddr_in syslog_sin ;
ret = vasprintf ( & s , format , ap ) ;
if ( ret = = - 1 ) {
return ;
}
2008-02-04 17:44:24 +11:00
switch ( this_log_level ) {
case DEBUG_EMERG :
level = LOG_EMERG ;
break ;
case DEBUG_ALERT :
level = LOG_ALERT ;
break ;
case DEBUG_CRIT :
level = LOG_CRIT ;
break ;
case DEBUG_ERR :
level = LOG_ERR ;
break ;
case DEBUG_WARNING :
level = LOG_WARNING ;
break ;
case DEBUG_NOTICE :
level = LOG_NOTICE ;
break ;
case DEBUG_INFO :
level = LOG_INFO ;
break ;
default :
level = LOG_DEBUG ;
break ;
}
2009-10-30 18:53:17 +11:00
len = offsetof ( struct syslog_message , message ) + strlen ( s ) + 1 ;
msg = malloc ( len ) ;
if ( msg = = NULL ) {
free ( s ) ;
return ;
}
msg - > level = level ;
msg - > len = strlen ( s ) ;
strcpy ( msg - > message , s ) ;
2009-10-30 19:39:11 +11:00
if ( syslogd_is_started = = 0 ) {
syslog ( msg - > level , " %s " , msg - > message ) ;
} else {
syslog_fd = socket ( AF_INET , SOCK_DGRAM , IPPROTO_UDP ) ;
if ( syslog_fd = = - 1 ) {
printf ( " Failed to create syslog socket \n " ) ;
free ( s ) ;
free ( msg ) ;
return ;
}
2009-10-30 18:53:17 +11:00
2009-10-30 19:39:11 +11:00
syslog_sin . sin_family = AF_INET ;
syslog_sin . sin_port = htons ( CTDB_PORT ) ;
syslog_sin . sin_addr . s_addr = htonl ( INADDR_LOOPBACK ) ;
ret = sendto ( syslog_fd , msg , len , 0 , & syslog_sin , sizeof ( syslog_sin ) ) ;
/* no point in checking here since we cant log an error */
2009-10-30 18:53:17 +11:00
2009-10-30 19:39:11 +11:00
close ( syslog_fd ) ;
}
2009-10-30 18:53:17 +11:00
free ( s ) ;
free ( msg ) ;
2008-01-16 22:03:01 +11:00
}
/*
log file logging function
*/
static void ctdb_logfile_log ( const char * format , va_list ap )
{
struct timeval t ;
char * s = NULL ;
struct tm * tm ;
char tbuf [ 100 ] ;
2008-01-29 14:31:51 +11:00
char * s2 = NULL ;
2009-05-20 12:08:13 +02:00
int ret ;
2008-01-16 22:03:01 +11:00
2009-05-20 12:08:13 +02:00
ret = vasprintf ( & s , format , ap ) ;
if ( ret = = - 1 ) {
2009-05-21 11:49:16 +10:00
const char * errstr = " vasprintf failed \n " ;
write ( log_state - > fd , errstr , strlen ( errstr ) ) ;
2009-05-20 12:08:13 +02:00
return ;
}
2008-01-16 22:03:01 +11:00
t = timeval_current ( ) ;
tm = localtime ( & t . tv_sec ) ;
strftime ( tbuf , sizeof ( tbuf ) - 1 , " %Y/%m/%d %H:%M:%S " , tm ) ;
2009-05-20 12:08:13 +02:00
ret = asprintf ( & s2 , " %s.%06u [%5u]: %s " ,
2008-01-29 14:31:51 +11:00
tbuf , ( unsigned ) t . tv_usec , ( unsigned ) getpid ( ) , s ) ;
free ( s ) ;
2009-05-20 12:08:13 +02:00
if ( ret = = - 1 ) {
2009-05-21 11:49:16 +10:00
const char * errstr = " asprintf failed \n " ;
write ( log_state - > fd , errstr , strlen ( errstr ) ) ;
2009-05-20 12:08:13 +02:00
return ;
}
2008-01-29 14:31:51 +11:00
if ( s2 ) {
write ( log_state - > fd , s2 , strlen ( s2 ) ) ;
2009-05-20 12:08:13 +02:00
free ( s2 ) ;
2008-01-29 14:31:51 +11:00
}
2008-01-16 22:03:01 +11:00
}
2009-07-21 09:48:10 +02:00
static void ctdb_logfile_log_add ( const char * format , va_list ap )
{
char * s = NULL ;
int ret ;
ret = vasprintf ( & s , format , ap ) ;
if ( ret = = - 1 ) {
const char * errstr = " vasprintf failed \n " ;
write ( log_state - > fd , errstr , strlen ( errstr ) ) ;
return ;
}
if ( s ) {
write ( log_state - > fd , s , strlen ( s ) ) ;
free ( s ) ;
}
}
2009-10-30 18:53:17 +11:00
2008-01-16 22:03:01 +11:00
/*
choose the logfile location
*/
int ctdb_set_logfile ( struct ctdb_context * ctdb , const char * logfile , bool use_syslog )
{
2009-05-20 12:08:13 +02:00
int ret ;
2009-10-30 18:53:17 +11:00
2008-01-16 22:03:01 +11:00
ctdb - > log = talloc_zero ( ctdb , struct ctdb_log_state ) ;
2009-05-20 11:47:34 +02:00
if ( ctdb - > log = = NULL ) {
printf ( " talloc_zero failed \n " ) ;
abort ( ) ;
}
2008-01-16 22:03:01 +11:00
2009-12-08 00:31:29 +10:30
ctdb - > log - > ctdb = ctdb ;
2008-01-16 22:03:01 +11:00
log_state = ctdb - > log ;
if ( use_syslog ) {
do_debug_v = ctdb_syslog_log ;
2009-07-21 09:50:56 +02:00
do_debug_add_v = ctdb_syslog_log ;
2008-01-16 22:03:01 +11:00
ctdb - > log - > use_syslog = true ;
} else if ( logfile = = NULL | | strcmp ( logfile , " - " ) = = 0 ) {
do_debug_v = ctdb_logfile_log ;
2009-07-21 09:48:10 +02:00
do_debug_add_v = ctdb_logfile_log_add ;
2008-01-16 22:03:01 +11:00
ctdb - > log - > fd = 1 ;
/* also catch stderr of subcommands to stdout */
2009-05-20 12:08:13 +02:00
ret = dup2 ( 1 , 2 ) ;
if ( ret = = - 1 ) {
printf ( " dup2 failed: %s \n " , strerror ( errno ) ) ;
abort ( ) ;
}
2008-01-16 22:03:01 +11:00
} else {
do_debug_v = ctdb_logfile_log ;
2009-07-21 09:48:10 +02:00
do_debug_add_v = ctdb_logfile_log_add ;
2008-01-16 22:03:01 +11:00
ctdb - > log - > fd = open ( logfile , O_WRONLY | O_APPEND | O_CREAT , 0666 ) ;
if ( ctdb - > log - > fd = = - 1 ) {
2009-05-20 11:47:34 +02:00
printf ( " Failed to open logfile %s \n " , logfile ) ;
2008-01-16 22:03:01 +11:00
abort ( ) ;
}
}
return 0 ;
}
2009-12-08 12:44:30 +10:30
/* Note that do_debug always uses the global log state. */
static void write_to_log ( struct ctdb_log_state * log ,
const char * buf , unsigned int len )
{
if ( script_log_level < = LogLevel ) {
do_debug ( " %*.*s \n " , len , len , buf ) ;
/* log it in the eventsystem as well */
if ( log - > logfn )
log - > logfn ( log - > buf , len , log - > logfn_private ) ;
}
}
2008-01-16 22:03:01 +11:00
/*
called when log data comes in from a child process
*/
static void ctdb_log_handler ( struct event_context * ev , struct fd_event * fde ,
uint16_t flags , void * private )
{
2009-12-08 00:31:29 +10:30
struct ctdb_log_state * log = talloc_get_type ( private , struct ctdb_log_state ) ;
2008-01-16 22:03:01 +11:00
char * p ;
2008-07-23 15:25:52 +10:00
int n ;
2008-01-16 22:03:01 +11:00
if ( ! ( flags & EVENT_FD_READ ) ) {
return ;
}
2009-12-08 00:31:29 +10:30
n = read ( log - > pfd , & log - > buf [ log - > buf_used ] ,
sizeof ( log - > buf ) - log - > buf_used ) ;
2008-07-23 15:25:52 +10:00
if ( n > 0 ) {
2009-12-08 00:31:29 +10:30
log - > buf_used + = n ;
2009-12-15 10:23:58 +11:00
} else if ( n = = 0 ) {
2009-12-15 19:04:52 +11:00
if ( log ! = log_state ) {
talloc_free ( log ) ;
}
2009-12-15 10:23:58 +11:00
return ;
2008-01-16 22:03:01 +11:00
}
2008-10-17 07:56:12 +11:00
this_log_level = script_log_level ;
2009-12-08 00:31:29 +10:30
while ( log - > buf_used > 0 & &
( p = memchr ( log - > buf , ' \n ' , log - > buf_used ) ) ! = NULL ) {
int n1 = ( p - log - > buf ) + 1 ;
2008-07-23 15:25:52 +10:00
int n2 = n1 - 1 ;
/* swallow \r from child processes */
2009-12-08 00:31:29 +10:30
if ( n2 > 0 & & log - > buf [ n2 - 1 ] = = ' \r ' ) {
2008-07-23 15:25:52 +10:00
n2 - - ;
2008-01-16 23:06:37 +11:00
}
2009-12-08 12:44:30 +10:30
write_to_log ( log , log - > buf , n2 ) ;
2009-12-08 00:31:29 +10:30
memmove ( log - > buf , p + 1 , sizeof ( log - > buf ) - n1 ) ;
log - > buf_used - = n1 ;
2008-01-16 22:03:01 +11:00
}
2008-07-23 15:25:52 +10:00
/* the buffer could have completely filled - unfortunately we have
no choice but to dump it out straight away */
2009-12-08 00:31:29 +10:30
if ( log - > buf_used = = sizeof ( log - > buf ) ) {
2009-12-08 12:44:30 +10:30
write_to_log ( log , log - > buf , log - > buf_used ) ;
2009-12-08 00:31:29 +10:30
log - > buf_used = 0 ;
2008-01-16 22:40:01 +11:00
}
2008-01-16 22:03:01 +11:00
}
2009-12-08 12:44:30 +10:30
static int log_context_destructor ( struct ctdb_log_state * log )
{
/* Flush buffer in case it wasn't \n-terminated. */
if ( log - > buf_used > 0 ) {
this_log_level = script_log_level ;
write_to_log ( log , log - > buf , log - > buf_used ) ;
}
return 0 ;
}
/*
fork ( ) , redirecting child output to logging and specified callback .
*/
struct ctdb_log_state * ctdb_fork_with_logging ( TALLOC_CTX * mem_ctx ,
struct ctdb_context * ctdb ,
void ( * logfn ) ( const char * , uint16_t , void * ) ,
void * logfn_private , pid_t * pid )
{
int p [ 2 ] ;
int old_stdout , old_stderr ;
int saved_errno ;
struct ctdb_log_state * log ;
log = talloc_zero ( mem_ctx , struct ctdb_log_state ) ;
CTDB_NO_MEMORY_NULL ( ctdb , log ) ;
log - > ctdb = ctdb ;
log - > logfn = logfn ;
log - > logfn_private = ( void * ) logfn_private ;
if ( pipe ( p ) ! = 0 ) {
DEBUG ( DEBUG_ERR , ( __location__ " Failed to setup for child logging pipe \n " ) ) ;
goto free_log ;
}
/* We'll fail if stderr/stdout not already open; it's simpler. */
old_stdout = dup ( STDOUT_FILENO ) ;
old_stderr = dup ( STDERR_FILENO ) ;
if ( dup2 ( p [ 1 ] , STDOUT_FILENO ) < 0 | | dup2 ( p [ 1 ] , STDERR_FILENO ) < 0 ) {
DEBUG ( DEBUG_ERR , ( __location__ " Failed to setup output for child \n " ) ) ;
goto close_pipe ;
}
close ( p [ 1 ] ) ;
* pid = fork ( ) ;
/* Child? */
if ( * pid = = 0 ) {
close ( old_stdout ) ;
close ( old_stderr ) ;
close ( p [ 0 ] ) ;
return log ;
}
2008-01-16 22:03:01 +11:00
2009-12-08 12:44:30 +10:30
saved_errno = errno ;
dup2 ( STDOUT_FILENO , old_stdout ) ;
dup2 ( STDERR_FILENO , old_stderr ) ;
close ( old_stdout ) ;
close ( old_stderr ) ;
/* We failed? */
if ( * pid < 0 ) {
DEBUG ( DEBUG_ERR , ( __location__ " fork failed for child process \n " ) ) ;
close ( p [ 0 ] ) ;
errno = saved_errno ;
goto free_log ;
}
log - > pfd = p [ 0 ] ;
talloc_set_destructor ( log , log_context_destructor ) ;
event_add_fd ( ctdb - > ev , log , log - > pfd , EVENT_FD_READ ,
ctdb_log_handler , log ) ;
return log ;
close_pipe :
close ( p [ 0 ] ) ;
close ( p [ 1 ] ) ;
free_log :
talloc_free ( log ) ;
return NULL ;
}
2008-01-16 22:03:01 +11:00
/*
setup for logging of child process stdout
*/
int ctdb_set_child_logging ( struct ctdb_context * ctdb )
{
int p [ 2 ] ;
2009-05-20 12:08:13 +02:00
int ret ;
2008-01-16 22:03:01 +11:00
if ( ctdb - > log - > fd = = 1 ) {
/* not needed for stdout logging */
return 0 ;
}
/* setup a pipe to catch IO from subprocesses */
if ( pipe ( p ) ! = 0 ) {
2008-02-04 20:07:15 +11:00
DEBUG ( DEBUG_ERR , ( __location__ " Failed to setup for child logging pipe \n " ) ) ;
2008-01-16 22:03:01 +11:00
return - 1 ;
}
2009-12-15 10:45:18 +11:00
event_add_fd ( ctdb - > ev , ctdb - > log , p [ 0 ] , EVENT_FD_READ ,
2009-12-08 00:31:29 +10:30
ctdb_log_handler , ctdb - > log ) ;
2008-01-16 22:03:01 +11:00
set_close_on_exec ( p [ 0 ] ) ;
ctdb - > log - > pfd = p [ 0 ] ;
2009-10-15 11:24:54 +11:00
DEBUG ( DEBUG_NOTICE , ( __location__ " Created PIPE FD:%d for logging \n " , p [ 0 ] ) ) ;
2008-01-16 22:03:01 +11:00
close ( 1 ) ;
close ( 2 ) ;
if ( p [ 1 ] ! = 1 ) {
2009-05-20 12:08:13 +02:00
ret = dup2 ( p [ 1 ] , 1 ) ;
if ( ret = = - 1 ) {
printf ( " dup2 failed: %s \n " , strerror ( errno ) ) ;
return - 1 ;
}
2008-01-16 22:03:01 +11:00
close ( p [ 1 ] ) ;
}
/* also catch stderr of subcommands to the log */
2009-05-20 12:08:13 +02:00
ret = dup2 ( 1 , 2 ) ;
if ( ret = = - 1 ) {
printf ( " dup2 failed: %s \n " , strerror ( errno ) ) ;
return - 1 ;
}
2008-01-16 22:03:01 +11:00
return 0 ;
}
2009-11-18 12:44:18 +11:00