2008-01-16 14:03:01 +03: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"
# include "../include/ctdb_private.h"
# include "system/syslog.h"
# include "system/time.h"
# include "system/filesys.h"
struct ctdb_log_state {
char * logfile ;
int fd , pfd ;
char buf [ 1024 ] ;
uint16_t buf_used ;
bool use_syslog ;
} ;
/* we need this global to eep the DEBUG() syntax */
static struct ctdb_log_state * log_state ;
/*
syslog logging function
*/
static void ctdb_syslog_log ( const char * format , va_list ap )
{
vsyslog ( LOG_NOTICE , format , ap ) ;
}
/*
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 06:31:51 +03:00
char * s2 = NULL ;
2008-01-16 14:03:01 +03:00
vasprintf ( & s , format , ap ) ;
t = timeval_current ( ) ;
tm = localtime ( & t . tv_sec ) ;
strftime ( tbuf , sizeof ( tbuf ) - 1 , " %Y/%m/%d %H:%M:%S " , tm ) ;
2008-01-29 06:31:51 +03:00
asprintf ( & s2 , " %s.%06u [%5u]: %s " ,
tbuf , ( unsigned ) t . tv_usec , ( unsigned ) getpid ( ) , s ) ;
free ( s ) ;
if ( s2 ) {
write ( log_state - > fd , s2 , strlen ( s2 ) ) ;
free ( s2 ) ;
}
2008-01-16 14:03:01 +03:00
}
/*
choose the logfile location
*/
int ctdb_set_logfile ( struct ctdb_context * ctdb , const char * logfile , bool use_syslog )
{
ctdb - > log = talloc_zero ( ctdb , struct ctdb_log_state ) ;
log_state = ctdb - > log ;
if ( use_syslog ) {
do_debug_v = ctdb_syslog_log ;
ctdb - > log - > use_syslog = true ;
} else if ( logfile = = NULL | | strcmp ( logfile , " - " ) = = 0 ) {
do_debug_v = ctdb_logfile_log ;
ctdb - > log - > fd = 1 ;
/* also catch stderr of subcommands to stdout */
dup2 ( 1 , 2 ) ;
} else {
do_debug_v = ctdb_logfile_log ;
ctdb - > log - > logfile = talloc_strdup ( ctdb , logfile ) ;
ctdb - > log - > fd = open ( logfile , O_WRONLY | O_APPEND | O_CREAT , 0666 ) ;
if ( ctdb - > log - > fd = = - 1 ) {
printf ( " Failed to open logfile %s \n " , ctdb - > logfile ) ;
abort ( ) ;
}
}
return 0 ;
}
/*
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 )
{
struct ctdb_context * ctdb = talloc_get_type ( private , struct ctdb_context ) ;
2008-01-16 14:40:01 +03:00
int n1 , n2 ;
2008-01-16 14:03:01 +03:00
char * p ;
if ( ! ( flags & EVENT_FD_READ ) ) {
return ;
}
2008-01-16 14:40:01 +03:00
n1 = read ( ctdb - > log - > pfd , & ctdb - > log - > buf [ ctdb - > log - > buf_used ] ,
2008-01-16 14:03:01 +03:00
sizeof ( ctdb - > log - > buf ) - ctdb - > log - > buf_used ) ;
2008-01-16 14:40:01 +03:00
if ( n1 > 0 ) {
ctdb - > log - > buf_used + = n1 ;
2008-01-16 14:03:01 +03:00
}
p = memchr ( ctdb - > log - > buf , ' \n ' , ctdb - > log - > buf_used ) ;
if ( ! p ) {
2008-01-16 15:06:37 +03:00
if ( ctdb - > log - > buf_used = = sizeof ( ctdb - > log - > buf ) ) {
do_debug ( " %*.*s \n " ,
( int ) ctdb - > log - > buf_used , ( int ) ctdb - > log - > buf_used , ctdb - > log - > buf ) ;
ctdb - > log - > buf_used = 0 ;
}
2008-01-16 14:03:01 +03:00
return ;
}
2008-01-16 14:40:01 +03:00
n1 = ( p - ctdb - > log - > buf ) + 1 ;
n2 = n1 - 1 ;
/* swallow \r from child processes */
if ( n2 > 0 & & ctdb - > log - > buf [ n2 - 1 ] = = ' \r ' ) {
n2 - - ;
}
do_debug ( " %*.*s \n " , n2 , n2 , ctdb - > log - > buf ) ;
memmove ( ctdb - > log - > buf , p + 1 , sizeof ( ctdb - > log - > buf ) - n1 ) ;
ctdb - > log - > buf_used - = n1 ;
2008-01-16 14:03:01 +03:00
}
/*
setup for logging of child process stdout
*/
int ctdb_set_child_logging ( struct ctdb_context * ctdb )
{
int p [ 2 ] ;
if ( ctdb - > log - > fd = = 1 ) {
/* not needed for stdout logging */
return 0 ;
}
/* setup a pipe to catch IO from subprocesses */
if ( pipe ( p ) ! = 0 ) {
DEBUG ( 0 , ( __location__ " Failed to setup for child logging pipe \n " ) ) ;
return - 1 ;
}
event_add_fd ( ctdb - > ev , ctdb , p [ 0 ] , EVENT_FD_READ ,
ctdb_log_handler , ctdb ) ;
set_close_on_exec ( p [ 0 ] ) ;
ctdb - > log - > pfd = p [ 0 ] ;
close ( 1 ) ;
close ( 2 ) ;
if ( p [ 1 ] ! = 1 ) {
dup2 ( p [ 1 ] , 1 ) ;
close ( p [ 1 ] ) ;
}
/* also catch stderr of subcommands to the log */
dup2 ( 1 , 2 ) ;
return 0 ;
}