Improve logging: version, date/time and debug flags

Also quieten start-up log messages for invoked commands, to reduce chatter
This commit is contained in:
Andrew Bettison 2013-03-12 19:18:30 +10:30
parent b17f5a6f63
commit da2c523bd6
3 changed files with 105 additions and 40 deletions

16
conf.c
View File

@ -78,16 +78,18 @@ static int reload(const char *path, int *resultp)
return -1; return -1;
if (cmp_meta(&meta, &conffile_meta) == 0) if (cmp_meta(&meta, &conffile_meta) == 0)
return 0; return 0;
if (conffile_meta.mtime != -1) if (conffile_meta.mtime != -1 && serverMode)
INFOF("config file %s -- detected new version", conffile_path()); INFOF("config file %s -- detected new version", conffile_path());
char *buf = NULL; char *buf = NULL;
if (meta.mtime == -1) if (meta.mtime == -1) {
INFOF("config file %s does not exist", path); if (serverMode)
else if (meta.size > CONFIG_FILE_MAX_SIZE) { INFOF("config file %s does not exist", path);
} else if (meta.size > CONFIG_FILE_MAX_SIZE) {
WHYF("config file %s is too big (%ld bytes exceeds limit %ld)", path, meta.size, CONFIG_FILE_MAX_SIZE); WHYF("config file %s is too big (%ld bytes exceeds limit %ld)", path, meta.size, CONFIG_FILE_MAX_SIZE);
return -1; return -1;
} else if (meta.size <= 0) { } else if (meta.size <= 0) {
INFOF("config file %s is zero size", path); if (serverMode)
INFOF("config file %s is zero size", path);
} else { } else {
FILE *f = fopen(path, "r"); FILE *f = fopen(path, "r");
if (f == NULL) { if (f == NULL) {
@ -112,7 +114,8 @@ static int reload(const char *path, int *resultp)
free(buf); free(buf);
return -1; return -1;
} }
INFOF("config file %s successfully read %ld bytes", path, (long) meta.size); if (serverMode)
INFOF("config file %s successfully read %ld bytes", path, (long) meta.size);
} }
conffile_meta = meta; conffile_meta = meta;
struct cf_om_node *new_root = NULL; struct cf_om_node *new_root = NULL;
@ -218,6 +221,7 @@ static int reload_and_parse(int permissive)
return WHYF("config file %s not loaded -- %s", conffile_path(), strbuf_str(b)); return WHYF("config file %s not loaded -- %s", conffile_path(), strbuf_str(b));
WARNF("config file %s loaded despite problems -- %s", conffile_path(), strbuf_str(b)); WARNF("config file %s loaded despite problems -- %s", conffile_path(), strbuf_str(b));
} }
logDebugFlags();
return 1; return 1;
} }

127
log.c
View File

@ -47,11 +47,21 @@ const struct __sourceloc __whence = __NOWHERE__;
static FILE *logfile = NULL; static FILE *logfile = NULL;
/* The logbuf is used to accumulate log messages before the log file is open and ready for /* The logbuf is used to accumulate log messages before the log file is open and ready for
writing. * writing.
*/ */
static char _log_buf[8192]; static char _log_buf[8192];
static struct strbuf logbuf = STRUCT_STRBUF_EMPTY; static struct strbuf logbuf = STRUCT_STRBUF_EMPTY;
/* Whether the software version has been logged in the current file yet.
*/
bool_t version_logged = 0;
struct tm last_tm;
/* The time stamp of the last logged message, used to detect when the date advances so that
* the date can be logged.
*/
struct tm last_tm;
#ifdef ANDROID #ifdef ANDROID
#include <android/log.h> #include <android/log.h>
#endif #endif
@ -81,19 +91,21 @@ static FILE *_open_logging()
} }
if (!logpath || !logpath[0]) { if (!logpath || !logpath[0]) {
logfile = stderr; logfile = stderr;
INFO("No logfile configured -- logging to stderr"); if (serverMode)
logMessage(LOG_LEVEL_INFO, __NOWHERE__, "No logfile configured -- logging to stderr");
} else { } else {
char path[1024]; char path[1024];
if (!FORM_SERVAL_INSTANCE_PATH(path, logpath)) { if (!FORM_SERVAL_INSTANCE_PATH(path, logpath)) {
logfile = stderr; logfile = stderr;
INFO("Logfile path overrun -- logging to stderr"); logMessage(LOG_LEVEL_WARN, __NOWHERE__, "Logfile path overrun -- logging to stderr");
} else if ((logfile = fopen(path, "a"))) { } else if ((logfile = fopen(path, "a"))) {
setlinebuf(logfile); setlinebuf(logfile);
INFOF("Logging to %s (fd %d)", path, fileno(logfile)); if (serverMode)
logMessage(LOG_LEVEL_INFO, __NOWHERE__, "Logging to %s (fd %d)", path, fileno(logfile));
} else { } else {
logfile = stderr; logfile = stderr;
WARNF_perror("fopen(%s)", path); WARNF_perror("fopen(%s)", path);
WARNF("Cannot append to %s -- falling back to stderr", path); logMessage(LOG_LEVEL_WARN, __NOWHERE__, "Cannot append to %s -- falling back to stderr", path);
} }
} }
} }
@ -135,46 +147,60 @@ static int _log_prepare(int level, struct __sourceloc whence)
tv.tv_sec = 0; tv.tv_sec = 0;
if (config.log.show_time) if (config.log.show_time)
gettimeofday(&tv, NULL); gettimeofday(&tv, NULL);
if (!version_logged)
logVersion();
_open_logging(); // Put initial INFO message at start of log file _open_logging(); // Put initial INFO message at start of log file
// No calls outside log.c from this point on. // No calls outside log.c from this point on.
if (strbuf_is_empty(&logbuf)) while (1) {
strbuf_init(&logbuf, _log_buf, sizeof _log_buf); if (strbuf_is_empty(&logbuf))
else if (strbuf_len(&logbuf)) strbuf_init(&logbuf, _log_buf, sizeof _log_buf);
strbuf_putc(&logbuf, '\n'); else if (strbuf_len(&logbuf))
strbuf_putc(&logbuf, '\n');
#ifndef ANDROID #ifndef ANDROID
const char *levelstr = "UNKWN:"; const char *levelstr = "UNKWN:";
switch (level) { switch (level) {
case LOG_LEVEL_FATAL: levelstr = "FATAL:"; break; case LOG_LEVEL_FATAL: levelstr = "FATAL:"; break;
case LOG_LEVEL_ERROR: levelstr = "ERROR:"; break; case LOG_LEVEL_ERROR: levelstr = "ERROR:"; break;
case LOG_LEVEL_INFO: levelstr = "INFO:"; break; case LOG_LEVEL_INFO: levelstr = "INFO:"; break;
case LOG_LEVEL_WARN: levelstr = "WARN:"; break; case LOG_LEVEL_WARN: levelstr = "WARN:"; break;
case LOG_LEVEL_DEBUG: levelstr = "DEBUG:"; break; case LOG_LEVEL_DEBUG: levelstr = "DEBUG:"; break;
}
strbuf_sprintf(&logbuf, "%-6.6s ", levelstr);
#endif
if (config.log.show_pid)
strbuf_sprintf(&logbuf, "[%5u] ", getpid());
if (config.log.show_time) {
if (tv.tv_sec == 0) {
strbuf_puts(&logbuf, "NOTIME______ ");
} else {
struct tm tm;
char buf[20];
if (strftime(buf, sizeof buf, "%T", localtime_r(&tv.tv_sec, &tm)) == 0)
strbuf_puts(&logbuf, "EMPTYTIME___ ");
else
strbuf_sprintf(&logbuf, "%s.%03u ", buf, tv.tv_usec / 1000);
} }
strbuf_sprintf(&logbuf, "%-6.6s", levelstr);
#endif
if (config.log.show_pid)
strbuf_sprintf(&logbuf, " [%5u]", getpid());
if (config.log.show_time) {
if (tv.tv_sec == 0) {
strbuf_puts(&logbuf, " NOTIME______");
} else {
struct tm tm;
localtime_r(&tv.tv_sec, &tm);
char buf[50];
if (strftime(buf, sizeof buf, "%T", &tm) == 0)
strbuf_puts(&logbuf, " EMPTYTIME___");
else
strbuf_sprintf(&logbuf, " %s.%03u", buf, tv.tv_usec / 1000);
if (tm.tm_mday != last_tm.tm_mday || tm.tm_mon != last_tm.tm_mon || tm.tm_year != last_tm.tm_year) {
if (strftime(buf, sizeof buf, "%F %T %z", &tm))
strbuf_puts(&logbuf, " Local date/time: ");
strbuf_puts(&logbuf, buf);
last_tm = tm;
continue;
}
last_tm = tm;
}
}
break;
} }
if (whence.file) { if (whence.file) {
strbuf_sprintf(&logbuf, "%s", _trimbuildpath(whence.file)); strbuf_sprintf(&logbuf, " %s", _trimbuildpath(whence.file));
if (whence.line) if (whence.line)
strbuf_sprintf(&logbuf, ":%u", whence.line); strbuf_sprintf(&logbuf, ":%u", whence.line);
if (whence.function) if (whence.function)
strbuf_sprintf(&logbuf, ":%s()", whence.function); strbuf_sprintf(&logbuf, ":%s()", whence.function);
strbuf_putc(&logbuf, ' '); strbuf_putc(&logbuf, ' ');
} else if (whence.function) { } else if (whence.function) {
strbuf_sprintf(&logbuf, "%s() ", whence.function); strbuf_sprintf(&logbuf, " %s() ", whence.function);
} }
strbuf_putc(&logbuf, ' '); strbuf_putc(&logbuf, ' ');
return 1; return 1;
@ -210,7 +236,11 @@ static void _log_internal(int level, struct strbuf *buf)
#else #else
FILE *logf = _open_logging(); FILE *logf = _open_logging();
if (logf) { if (logf) {
fprintf(logf, "%s%s%s", strbuf_str(buf), strbuf_len(buf) ? "\n" : "", strbuf_overrun(buf) ? "LOG OVERRUN\n" : ""); fprintf(logf, "%s%s%s",
strbuf_len(buf) ? strbuf_str(buf) : "",
strbuf_len(buf) ? "\n" : "",
strbuf_overrun(buf) ? "LOG OVERRUN\n" : ""
);
strbuf_reset(buf); strbuf_reset(buf);
} }
#endif #endif
@ -282,6 +312,35 @@ void vlogMessage(int level, struct __sourceloc whence, const char *fmt, va_list
} }
} }
void logVersion()
{
version_logged = 1;
logMessage(LOG_LEVEL_INFO, __NOWHERE__, "Serval DNA version: %s", version_servald);
}
void logDebugFlags()
{
struct cf_om_node *debug_root = NULL;
cf_fmt_config_debug(&debug_root, &config.debug);
strbuf b = strbuf_alloca(1024);
struct cf_om_iterator it;
for (cf_om_iter_start(&it, debug_root); it.node; cf_om_iter_next(&it))
if (it.node->text) {
bool_t val = 0;
if (cf_opt_boolean(&val, it.node->text) != CFOK || val) {
if (strbuf_len(b))
strbuf_puts(b, ", ");
strbuf_puts(b, it.node->fullkey);
if (!val) {
strbuf_putc(b, '=');
strbuf_puts(b, it.node->text);
}
}
}
if (strbuf_len(b))
logMessage(LOG_LEVEL_INFO, __NOWHERE__, "Debug options: %s", strbuf_str(b));
}
int logDump(int level, struct __sourceloc whence, char *name, const unsigned char *addr, size_t len) int logDump(int level, struct __sourceloc whence, char *name, const unsigned char *addr, size_t len)
{ {
char buf[100]; char buf[100];

2
log.h
View File

@ -92,6 +92,8 @@ void logArgv(int level, struct __sourceloc whence, const char *label, int argc,
void logString(int level, struct __sourceloc whence, const char *str); void logString(int level, struct __sourceloc whence, const char *str);
void logMessage(int level, struct __sourceloc whence, const char *fmt, ...); void logMessage(int level, struct __sourceloc whence, const char *fmt, ...);
void vlogMessage(int level, struct __sourceloc whence, const char *fmt, va_list); void vlogMessage(int level, struct __sourceloc whence, const char *fmt, va_list);
void logVersion();
void logDebugFlags();
int logDump(int level, struct __sourceloc whence, char *name, const unsigned char *addr, size_t len); int logDump(int level, struct __sourceloc whence, char *name, const unsigned char *addr, size_t len);
ssize_t get_self_executable_path(char *buf, size_t len); ssize_t get_self_executable_path(char *buf, size_t len);
int log_backtrace(struct __sourceloc whence); int log_backtrace(struct __sourceloc whence);