2012-08-08 09:41:01 +02:00
# include "daemon-server.h"
# include "daemon-log.h"
# include <syslog.h>
2012-08-17 01:12:10 +02:00
# include <assert.h>
2012-08-08 09:41:01 +02:00
struct backend {
int id ;
void ( * log ) ( log_state * s , void * * state , int type , const char * message ) ;
} ;
static void log_syslog ( log_state * s , void * * state , int type , const char * message )
{
2012-10-12 10:15:30 +02:00
int prio ;
2012-08-08 09:41:01 +02:00
if ( ! * state ) { /* initialize */
* state = ( void * ) 1 ;
openlog ( s - > name , LOG_PID , LOG_DAEMON ) ;
}
2012-10-12 10:15:30 +02:00
2012-08-08 09:41:01 +02:00
switch ( type ) {
case DAEMON_LOG_INFO : prio = LOG_INFO ; break ;
case DAEMON_LOG_WARN : prio = LOG_WARNING ; break ;
2015-03-10 14:35:29 +01:00
case DAEMON_LOG_ERROR : prio = LOG_ERR ; break ;
2012-08-08 09:41:01 +02:00
case DAEMON_LOG_FATAL : prio = LOG_CRIT ; break ;
2012-10-12 10:15:30 +02:00
default : prio = LOG_DEBUG ; break ;
2012-08-08 09:41:01 +02:00
}
syslog ( prio , " %s " , message ) ;
}
static void log_stderr ( log_state * s , void * * state , int type , const char * message )
{
2012-10-12 10:15:30 +02:00
const char * prefix ;
2012-08-08 09:41:01 +02:00
switch ( type ) {
case DAEMON_LOG_INFO : prefix = " I: " ; break ;
case DAEMON_LOG_WARN : prefix = " W: " ; break ;
2012-10-12 10:15:30 +02:00
case DAEMON_LOG_ERROR : /* fall through */
2012-08-08 09:41:01 +02:00
case DAEMON_LOG_FATAL : prefix = " E: " ; break ;
2012-10-12 10:15:30 +02:00
default : prefix = " " ; break ;
2012-08-08 09:41:01 +02:00
}
fprintf ( stderr , " %s%s \n " , prefix , message ) ;
}
struct backend backend [ ] = {
{ DAEMON_LOG_OUTLET_SYSLOG , log_syslog } ,
{ DAEMON_LOG_OUTLET_STDERR , log_stderr } ,
{ 0 , 0 }
} ;
void daemon_log ( log_state * s , int type , const char * message ) {
int i = 0 ;
while ( backend [ i ] . id ) {
2013-11-25 13:42:30 +01:00
if ( ( int ) ( s - > log_config [ type ] & backend [ i ] . id ) = = backend [ i ] . id )
2012-08-08 09:41:01 +02:00
backend [ i ] . log ( s , & s - > backend_state [ i ] , type , message ) ;
+ + i ;
}
}
2012-10-08 18:34:33 +02:00
static int _type_interesting ( log_state * s , int type ) {
int i = 0 ;
while ( backend [ i ] . id ) {
2013-11-25 13:42:30 +01:00
if ( ( int ) ( s - > log_config [ type ] & backend [ i ] . id ) = = backend [ i ] . id )
2012-10-08 18:34:33 +02:00
return 1 ;
+ + i ;
}
return 0 ;
}
2012-08-08 09:41:01 +02:00
void daemon_logf ( log_state * s , int type , const char * fmt , . . . ) {
2012-08-17 00:46:00 +02:00
char * buf ;
2012-08-08 09:41:01 +02:00
va_list ap ;
2012-08-17 00:46:00 +02:00
2012-08-08 09:41:01 +02:00
va_start ( ap , fmt ) ;
2012-08-17 00:46:00 +02:00
if ( dm_vasprintf ( & buf , fmt , ap ) > = 0 ) {
daemon_log ( s , type , buf ) ;
dm_free ( buf ) ;
} /* else return_0 */
va_end ( ap ) ;
2012-08-08 09:41:01 +02:00
}
struct log_line_baton {
log_state * s ;
int type ;
const char * prefix ;
} ;
static int _log_line ( const char * line , void * baton ) {
struct log_line_baton * b = baton ;
daemon_logf ( b - > s , b - > type , " %s%s " , b - > prefix , line ) ;
return 0 ;
}
void daemon_log_cft ( log_state * s , int type , const char * prefix , const struct dm_config_node * n )
{
2012-10-12 10:15:30 +02:00
struct log_line_baton b = { . s = s , . type = type , . prefix = prefix } ;
2012-10-08 18:34:33 +02:00
if ( ! _type_interesting ( s , type ) )
return ;
2012-12-13 19:55:33 +01:00
( void ) dm_config_write_node ( n , & _log_line , & b ) ;
2012-08-08 09:41:01 +02:00
}
void daemon_log_multi ( log_state * s , int type , const char * prefix , const char * msg )
{
2012-10-12 10:15:30 +02:00
struct log_line_baton b = { . s = s , . type = type , . prefix = prefix } ;
char * buf ;
char * pos ;
2012-10-08 18:34:33 +02:00
if ( ! _type_interesting ( s , type ) )
return ;
2012-10-12 10:15:30 +02:00
buf = dm_strdup ( msg ) ;
pos = buf ;
2012-08-08 09:41:01 +02:00
if ( ! buf )
return ; /* _0 */
while ( pos ) {
char * next = strchr ( pos , ' \n ' ) ;
if ( next )
* next = 0 ;
_log_line ( pos , & b ) ;
pos = next ? next + 1 : 0 ;
}
2012-10-08 17:11:43 +02:00
dm_free ( buf ) ;
2012-08-08 09:41:01 +02:00
}
void daemon_log_enable ( log_state * s , int outlet , int type , int enable )
{
assert ( type < 32 ) ;
if ( enable )
s - > log_config [ type ] | = outlet ;
else
s - > log_config [ type ] & = ~ outlet ;
}
static int _parse_one ( log_state * s , int outlet , const char * type , int enable )
{
int i ;
if ( ! strcmp ( type , " all " ) )
for ( i = 0 ; i < 32 ; + + i )
daemon_log_enable ( s , outlet , i , enable ) ;
else if ( ! strcmp ( type , " wire " ) )
daemon_log_enable ( s , outlet , DAEMON_LOG_WIRE , enable ) ;
else if ( ! strcmp ( type , " debug " ) )
daemon_log_enable ( s , outlet , DAEMON_LOG_DEBUG , enable ) ;
else
return 0 ;
return 1 ;
}
int daemon_log_parse ( log_state * s , int outlet , const char * types , int enable )
{
2012-10-15 11:02:15 +02:00
char * buf ;
char * pos ;
2012-08-08 09:41:01 +02:00
2012-10-15 11:02:15 +02:00
if ( ! types | | ! types [ 0 ] )
return 1 ;
if ( ! ( buf = dm_strdup ( types ) ) )
2012-08-08 09:41:01 +02:00
return 0 ;
2012-10-15 11:02:15 +02:00
pos = buf ;
2012-08-08 09:41:01 +02:00
while ( pos ) {
char * next = strchr ( pos , ' , ' ) ;
if ( next )
* next = 0 ;
2012-08-16 20:10:35 +02:00
if ( ! _parse_one ( s , outlet , pos , enable ) ) {
dm_free ( buf ) ;
2012-08-08 09:41:01 +02:00
return 0 ;
2012-08-16 20:10:35 +02:00
}
2012-08-08 09:41:01 +02:00
pos = next ? next + 1 : 0 ;
}
2012-08-16 20:10:35 +02:00
dm_free ( buf ) ;
2012-08-08 09:41:01 +02:00
return 1 ;
}