Newer
Older
*
* Distributed under the terms of the GNU General Public License (GPL) Version 2
*
* Logging routines
*
*/
#include <signal.h>
#include <stdarg.h>
#include <stdio.h>
#include <unistd.h>
#include <time.h>
#include <asterisk/logger.h>
#include <asterisk/options.h>
#include <string.h>
#include <stdlib.h>
#include <errno.h>
#include <pthread.h>
#include <sys/stat.h>
#include <syslog.h>
static ast_mutex_t msglist_lock = AST_MUTEX_INITIALIZER;
static ast_mutex_t loglock = AST_MUTEX_INITIALIZER;
static struct msglist {
char *msg;
struct msglist *next;
} *list = NULL, *last = NULL;
struct logfile {
char fn[256];
int logflags;
FILE *f;
int facility; /* syslog */
struct logfile *next;
};
static struct logfile *logfiles = NULL;
static int msgcnt = 0;
static FILE *eventlog = NULL;
static char *levels[] = {
"DEBUG",
"EVENT",
"NOTICE",
"WARNING",
"ERROR"
};
static int colors[] = {
COLOR_BRGREEN,
COLOR_BRBLUE,
COLOR_YELLOW,
COLOR_BRRED,
COLOR_RED
};
static int make_components(char *s, int lineno)
{
char *w;
int res = 0;
char *stringp=NULL;
stringp=s;
w = strsep(&stringp, ",");
while(w) {
while(*w && (*w < 33))
w++;
if (!strcasecmp(w, "debug"))
res |= (1 << 0);
else if (!strcasecmp(w, "notice"))
res |= (1 << 2);
else if (!strcasecmp(w, "warning"))
res |= (1 << 3);
else if (!strcasecmp(w, "error"))
res |= (1 << 4);
else {
fprintf(stderr, "Logfile Warning: Unknown keyword '%s' at line %d of logger.conf\n", w, lineno);
}
}
return res;
}
static struct logfile *make_logfile(char *fn, char *components, int lineno)
{
struct logfile *f;
char tmp[256];
if (!strlen(fn))
return NULL;
f = malloc(sizeof(struct logfile));
if (f) {
strncpy(f->fn, fn, sizeof(f->fn) - 1);
if (!strcasecmp(fn, "ignore")) {
f->f = NULL;
} else if (!strcasecmp(fn, "console")) {
f->f = stdout;
} else if (!strcasecmp(fn, "syslog")) {
f->f = NULL;
f->facility = LOG_LOCAL0;
} else {
if (fn[0] == '/')
strncpy(tmp, fn, sizeof(tmp) - 1);
else
snprintf(tmp, sizeof(tmp), "%s/%s", (char *)ast_config_AST_LOG_DIR, fn);
f->f = fopen(tmp, "a");
if (!f->f) {
/* Can't log here, since we're called with a lock */
fprintf(stderr, "Logger Warning: Unable to open log file '%s': %s\n", tmp, strerror(errno));
}
}
f->logflags = make_components(components, lineno);
}
return f;
}
static void init_logger_chain(void)
{
struct logfile *f, *cur;
struct ast_config *cfg;
struct ast_variable *var;
ast_mutex_lock(&loglock);
/* Free anything that is here */
f = logfiles;
while(f) {
cur = f->next;
if (f->f && (f->f != stdout) && (f->f != stderr))
fclose(f->f);
free(f);
f = cur;
}
logfiles = NULL;
ast_mutex_unlock(&loglock);
ast_mutex_lock(&loglock);
/* If no config file, we're fine */
if (!cfg) {
ast_mutex_unlock(&loglock);
return;
}
var = ast_variable_browse(cfg, "logfiles");
while(var) {
f = make_logfile(var->name, var->value, var->lineno);
if (f) {
f->next = logfiles;
logfiles = f;
}
var = var->next;
}
if (!logfiles) {
/* Gotta have at least one. We'll make a NULL one */
logfiles = make_logfile("ignore", "", -1);
}
ast_mutex_unlock(&loglock);
int reload_logger(int rotate)
char old[AST_CONFIG_MAX_PATH];
char new[AST_CONFIG_MAX_PATH];
struct logfile *f;
int x;
if (eventlog)
fclose(eventlog);
else
rotate = 0;
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
snprintf(old, sizeof(old), "%s/%s", (char *)ast_config_AST_LOG_DIR, EVENTLOG);
for(x=0;;x++) {
snprintf(new, sizeof(new), "%s/%s.%d", (char *)ast_config_AST_LOG_DIR, EVENTLOG,x);
eventlog = fopen((char *)new, "r");
if(eventlog)
fclose(eventlog);
else
break;
}
if(rotate) {
/* do it */
if(! link(old,new))
unlink(old);
strcpy(tmp,old);
}
f = logfiles;
while(f) {
if (f->f && (f->f != stdout) && (f->f != stderr)) {
fclose(f->f);
snprintf(old, sizeof(old), "%s/%s", (char *)ast_config_AST_LOG_DIR,f->fn);
for(x=0;;x++) {
snprintf(new, sizeof(new), "%s/%s.%d", (char *)ast_config_AST_LOG_DIR,f->fn,x);
eventlog = fopen((char *)new, "r");
if(eventlog)
fclose(eventlog);
else
break;
}
if(rotate) {
/* do it */
if(! link(old,new))
unlink(old);
f->f = fopen((char *)old, "a");
}
}
f = f->next;
}
ast_log(LOG_EVENT, "Restarted Asterisk Event Logger\n");
ast_verbose("Asterisk Event Logger restarted\n");
return 0;
} else
ast_log(LOG_ERROR, "Unable to create event log: %s\n", strerror(errno));
static int handle_logger_reload(int fd, int argc, char *argv[])
{
if(reload_logger(0))
{
ast_cli(fd, "Failed to reloadthe logger\n");
return RESULT_FAILURE;
}
else
return RESULT_SUCCESS;
}
static int handle_logger_rotate(int fd, int argc, char *argv[])
{
if(reload_logger(1))
{
ast_cli(fd, "Failed to reloadthe logger\n");
return RESULT_FAILURE;
}
else
return RESULT_SUCCESS;
}
static struct verb {
void (*verboser)(const char *string, int opos, int replacelast, int complete);
struct verb *next;
} *verboser = NULL;
static char logger_reload_help[] =
"Usage: logger reload\n"
" Reopens the log files. Use after a rotating the log files\n";
static char logger_rotate_help[] =
"Usage: logger reload\n"
" Rotates and Reopens the log files.\n";
static struct ast_cli_entry reload_logger_cli =
{ { "logger", "reload", NULL },
handle_logger_reload, "Reopens the log files",
logger_reload_help };
static struct ast_cli_entry rotate_logger_cli =
{ { "logger", "rotate", NULL },
handle_logger_rotate, "Reopens the log files",
logger_rotate_help };
static int handle_SIGXFSZ(int sig) {
reload_logger(1);
ast_log(LOG_EVENT,"Rotated Logs Per SIGXFSZ\n");
if (option_verbose)
ast_verbose("Rotated Logs Per SIGXFSZ\n");
return 0;
}
/* auto rotate if sig SIGXFSZ comes a-knockin */
(void) signal(SIGXFSZ,(void *) handle_SIGXFSZ);
/* register the relaod logger cli command */
ast_cli_register(&reload_logger_cli);
ast_cli_register(&rotate_logger_cli);
mkdir((char *)ast_config_AST_LOG_DIR, 0755);
snprintf(tmp, sizeof(tmp), "%s/%s", (char *)ast_config_AST_LOG_DIR, EVENTLOG);
eventlog = fopen((char *)tmp, "a");
ast_log(LOG_EVENT, "Started Asterisk Event Logger\n");
ast_verbose("Asterisk Event Logger Started %s\n",(char *)tmp);
return 0;
} else
ast_log(LOG_ERROR, "Unable to create event log: %s\n", strerror(errno));
init_logger_chain();
extern void ast_log(int level, const char *file, int line, const char *function, const char *fmt, ...)
char tmp[80];
char tmp2[80];
char tmp3[80];
char tmp4[80];
char linestr[80];
if (!option_verbose && !option_debug && (!level)) {
ast_mutex_lock(&loglock);
localtime_r(&t,&tm);
if (&tm) {
/* Log events into the event log file, with a different format */
strftime(date, sizeof(date), "%b %e %T", &tm);
fprintf(eventlog, "%s asterisk[%d]: ", date, getpid());
/** Cannot use ast_log() from locked section of ast_log()!
ast_log(LOG_WARNING, "Unable to retrieve local time?\n"); **/
fprintf(stderr, "ast_log: Unable to retrieve local time for %ld?\n", (long)t);
if (f->logflags & (1 << level) && f->facility) {
time(&t);
localtime_r(&t,&tm);
strftime(date, sizeof(date), "%b %e %T", &tm);
openlog("asterisk_pbx",LOG_PID,f->facility);
syslog(LOG_INFO|f->facility,"%s %s[%ld]: File %s, Line %d (%s): ",date,
levels[level], (long)pthread_self(), file, line, function);
closelog();
}
else if (f->logflags & (1 << level) && f->f) {
if ((f->f != stdout) && (f->f != stderr)) {
time(&t);
localtime_r(&t,&tm);
strftime(date, sizeof(date), "%b %e %T", &tm);
fprintf(f->f, "%s %s[%ld]: File %s, Line %d (%s): ", date, levels[level], (long)pthread_self(), file, line, function);
sprintf(linestr, "%d", line);
fprintf(f->f, "%s[%ld]: File %s, Line %s (%s): ",
term_color(tmp, levels[level], colors[level], 0, sizeof(tmp)),
term_color(tmp2, file, COLOR_BRWHITE, 0, sizeof(tmp2)),
term_color(tmp3, linestr, COLOR_BRWHITE, 0, sizeof(tmp3)),
term_color(tmp4, function, COLOR_BRWHITE, 0, sizeof(tmp4)));
fprintf(stdout, "%s[%ld]: File %s, Line %d (%s): ", levels[level], (long)pthread_self(), file, line, function);
ast_mutex_unlock(&loglock);
extern void ast_verbose(const char *fmt, ...)
static int pos = 0, opos;
static int replacelast = 0, complete;
struct msglist *m;
struct verb *v;
va_list ap;
va_start(ap, fmt);
ast_mutex_lock(&msglist_lock);
453
454
455
456
457
458
459
460
461
462
463
464
465
466
467
468
469
470
471
472
473
474
475
476
477
478
479
480
481
vsnprintf(stuff + pos, sizeof(stuff) - pos, fmt, ap);
opos = pos;
pos = strlen(stuff);
if (fmt[strlen(fmt)-1] == '\n')
complete = 1;
else
complete=0;
if (complete) {
if (msgcnt < MAX_MSG_QUEUE) {
/* Allocate new structure */
m = malloc(sizeof(struct msglist));
msgcnt++;
} else {
/* Recycle the oldest entry */
m = list;
list = list->next;
free(m->msg);
}
if (m) {
m->msg = strdup(stuff);
if (m->msg) {
if (last)
last->next = m;
else
list = m;
m->next = NULL;
last = m;
} else {
msgcnt--;
ast_log(LOG_ERROR, "Out of memory\n");
free(m);
}
}
}
if (verboser) {
v = verboser;
while(v) {
v->verboser(stuff, opos, replacelast, complete);
v = v->next;
}
if (fmt[strlen(fmt)-1] != '\n')
replacelast = 1;
else
replacelast = pos = 0;
va_end(ap);
ast_mutex_unlock(&msglist_lock);
int ast_verbose_dmesg(void (*v)(const char *string, int opos, int replacelast, int complete))
ast_mutex_lock(&msglist_lock);
while(m) {
/* Send all the existing entries that we have queued (i.e. they're likely to have missed) */
v(m->msg, 0, 0, 1);
m = m->next;
}
ast_mutex_unlock(&msglist_lock);
int ast_register_verbose(void (*v)(const char *string, int opos, int replacelast, int complete))
{
struct msglist *m;
struct verb *tmp;
/* XXX Should be more flexible here, taking > 1 verboser XXX */
if ((tmp = malloc(sizeof (struct verb)))) {
tmp->verboser = v;
ast_mutex_lock(&msglist_lock);
tmp->next = verboser;
verboser = tmp;
m = list;
while(m) {
/* Send all the existing entries that we have queued (i.e. they're likely to have missed) */
v(m->msg, 0, 0, 1);
m = m->next;
}
ast_mutex_unlock(&msglist_lock);
int ast_unregister_verbose(void (*v)(const char *string, int opos, int replacelast, int complete))
{
int res = -1;
struct verb *tmp, *tmpl=NULL;
ast_mutex_lock(&msglist_lock);
tmp = verboser;
while(tmp) {
if (tmp->verboser == v) {
if (tmpl)
tmpl->next = tmp->next;
else
verboser = tmp->next;
break;
}
tmpl = tmp;
tmp = tmp->next;
}
if (tmp)
res = 0;
ast_mutex_unlock(&msglist_lock);