server/src/util/log.c

320 lines
8 KiB
C
Raw Normal View History

2010-08-08 10:06:34 +02:00
/* vi: set ts=2:
+-------------------+ Christian Schlittchen <corwin@amber.kn-bremen.de>
| | Enno Rehling <enno@eressea.de>
| Eressea PBEM host | Katja Zedel <katze@felidae.kn-bremen.de>
| (c) 1998 - 2003 | Henning Peters <faroul@beyond.kn-bremen.de>
| | Ingo Wilken <Ingo.Wilken@informatik.uni-oldenburg.de>
+-------------------+ Stefan Reich <reich@halbling.de>
2012-06-04 03:41:07 +02:00
This program may not be used, modified or distributed
2010-08-08 10:06:34 +02:00
without prior permission by the authors of Eressea.
*/
#include <platform.h>
#include "log.h"
#include "unicode.h"
2012-06-04 03:41:07 +02:00
#include <assert.h>
2010-08-08 10:06:34 +02:00
#include <stdlib.h>
#include <string.h>
#include <stdio.h>
#include <stdarg.h>
#include <time.h>
/* TODO: set from external function */
int log_flags = LOG_FLUSH | LOG_CPERROR | LOG_CPWARNING | LOG_CPDEBUG;
2012-05-16 07:21:59 +02:00
int log_stderr = LOG_FLUSH | LOG_CPERROR | LOG_CPWARNING;
2010-08-08 10:06:34 +02:00
#ifdef STDIO_CP
static int stdio_codepage = STDIO_CP;
#else
static int stdio_codepage = 0;
#endif
2011-03-07 08:02:35 +01:00
static FILE *logfile;
2010-08-08 10:06:34 +02:00
2011-03-07 08:02:35 +01:00
#define MAXLENGTH 4096 /* because I am lazy, CP437 output is limited to this many chars */
#define LOG_MAXBACKUPS 5
2011-03-07 08:02:35 +01:00
void log_flush(void)
2010-08-08 10:06:34 +02:00
{
if (logfile) fflush(logfile);
2010-08-08 10:06:34 +02:00
}
2011-03-07 08:02:35 +01:00
void log_puts(const char *str)
2010-08-08 10:06:34 +02:00
{
fflush(stdout);
2012-05-16 07:21:59 +02:00
if (logfile) {
fputs(str, logfile);
}
2010-08-08 10:06:34 +02:00
}
static int
2011-03-07 08:02:35 +01:00
cp_convert(const char *format, char *buffer, size_t length, int codepage)
2010-08-08 10:06:34 +02:00
{
/* when console output on MSDOS, convert to codepage */
2011-03-07 08:02:35 +01:00
const char *input = format;
char *pos = buffer;
2010-08-08 10:06:34 +02:00
2011-03-07 08:02:35 +01:00
while (pos + 1 < buffer + length && *input) {
2010-08-08 10:06:34 +02:00
size_t length = 0;
int result = 0;
2011-03-07 08:02:35 +01:00
if (codepage == 437) {
2010-08-08 10:06:34 +02:00
result = unicode_utf8_to_cp437(pos, input, &length);
2011-03-07 08:02:35 +01:00
} else if (codepage == 1252) {
2010-08-08 10:06:34 +02:00
result = unicode_utf8_to_cp1252(pos, input, &length);
}
2011-03-07 08:02:35 +01:00
if (result != 0) {
*pos = 0; /* just in case caller ignores our return value */
2010-08-08 10:06:34 +02:00
return result;
}
++pos;
2011-03-07 08:02:35 +01:00
input += length;
2010-08-08 10:06:34 +02:00
}
*pos = 0;
return 0;
}
void log_rotate(const char *filename, int maxindex)
{
int n;
if (_access(filename, 4)==0) {
char buffer[2][MAX_PATH];
int src = 1;
assert(strlen(filename)<sizeof(buffer[0])-4);
for(n=0;n<maxindex;++n) {
sprintf(buffer[0], "%s.%d", filename, n);
if (_access(filename, 0)!=0) {
break;
}
}
if (_access(buffer[0], 0)==0) {
unlink(buffer[0]);
}
while(n--) {
int dst = 1-src;
sprintf(buffer[src], "%s.%d", filename, n);
rename(buffer[src], buffer[dst]);
src=dst;
}
rename(filename, buffer[1-src]);
}
}
2011-03-07 08:02:35 +01:00
void log_open(const char *filename)
2010-08-08 10:06:34 +02:00
{
if (logfile) {
2011-03-07 08:02:35 +01:00
log_close();
}
log_rotate(filename, LOG_MAXBACKUPS);
2010-08-08 10:06:34 +02:00
logfile = fopen(filename, "a");
if (logfile) {
/* Get UNIX-style time and display as number and string. */
time_t ltime;
2011-03-07 08:02:35 +01:00
time(&ltime);
2012-05-16 07:21:59 +02:00
fprintf(logfile, "===\n=== Logfile started at %s===\n", ctime(&ltime));
2010-08-08 10:06:34 +02:00
}
}
2011-03-07 08:02:35 +01:00
void log_close(void)
2010-08-08 10:06:34 +02:00
{
2011-03-07 08:02:35 +01:00
if (!logfile || logfile == stderr || logfile == stdout)
return;
2010-08-08 10:06:34 +02:00
if (logfile) {
/* Get UNIX-style time and display as number and string. */
time_t ltime;
2011-03-07 08:02:35 +01:00
time(&ltime);
2012-05-16 07:21:59 +02:00
fprintf(logfile, "===\n=== Logfile closed at %s===\n\n", ctime(&ltime));
fclose(logfile);
2010-08-08 10:06:34 +02:00
}
logfile = 0;
}
2011-03-07 08:02:35 +01:00
static int check_dupe(const char *format, const char *type)
2010-08-08 10:06:34 +02:00
{
2011-03-07 08:02:35 +01:00
static const char *last_type; /* STATIC_XCALL: used across calls */
2010-08-08 10:06:34 +02:00
static char last_message[32]; /* STATIC_XCALL: used across calls */
2011-03-07 08:02:35 +01:00
static int dupes = 0; /* STATIC_XCALL: used across calls */
if (strncmp(last_message, format, sizeof(last_message)) == 0) {
2010-08-08 10:06:34 +02:00
++dupes;
return 1;
}
if (dupes) {
if (log_flags & LOG_CPERROR) {
2011-03-07 08:02:35 +01:00
fprintf(stderr, "%s: last message repeated %d times\n", last_type,
dupes + 1);
2010-08-08 10:06:34 +02:00
}
dupes = 0;
}
strncpy(last_message, format, sizeof(last_message));
last_type = type;
return 0;
}
2012-05-16 07:21:59 +02:00
static void _log_write(FILE * stream, int codepage, const char * prefix, const char *format, va_list args)
{
2012-05-16 07:21:59 +02:00
if (stream) {
fprintf(stream, "%s: ", prefix);
if (codepage) {
char buffer[MAXLENGTH];
char converted[MAXLENGTH];
2012-06-04 03:41:07 +02:00
2012-05-16 07:21:59 +02:00
vsnprintf(buffer, sizeof(buffer), format, args);
if (cp_convert(buffer, converted, MAXLENGTH, codepage) == 0) {
fputs(converted, stream);
} else {
2012-05-16 07:21:59 +02:00
/* fall back to non-converted output */
vfprintf(stream, format, args);
}
2012-05-16 07:21:59 +02:00
} else {
vfprintf(stream, format, args);
}
2012-05-16 07:21:59 +02:00
}
}
static void _log_writeln(FILE * stream, int codepage, const char * prefix, const char *format, va_list args)
{
size_t len = strlen(format);
_log_write(stream, codepage, prefix, format, args);
if (format[len-1]!='\n') {
fputc('\n', stream);
}
}
void log_debug(const char *format, ...)
2012-05-16 07:21:59 +02:00
{
const char * prefix = "DEBUG";
const int mask = LOG_CPDEBUG;
/* write to the logfile, always */
if (logfile && (log_flags & mask)) {
va_list args;
va_start(args, format);
_log_writeln(logfile, 0, prefix, format, args);
2012-05-16 07:21:59 +02:00
va_end(args);
}
/* write to stderr, if that's not the logfile already */
if (logfile!=stderr && (log_stderr & mask)) {
int dupe = check_dupe(format, prefix);
if (!dupe) {
va_list args;
va_start(args, format);
_log_writeln(stderr, stdio_codepage, prefix, format, args);
2012-05-16 07:21:59 +02:00
va_end(args);
}
}
2012-05-16 07:21:59 +02:00
if (log_flags & LOG_FLUSH) {
log_flush();
}
}
void log_warning(const char *format, ...)
2010-08-08 10:06:34 +02:00
{
2012-05-16 07:21:59 +02:00
const char * prefix = "WARNING";
const int mask = LOG_CPWARNING;
2010-08-08 10:06:34 +02:00
2012-05-16 07:21:59 +02:00
/* write to the logfile, always */
if (logfile && (log_flags & mask)) {
va_list args;
va_start(args, format);
_log_writeln(logfile, 0, prefix, format, args);
2014-06-17 04:12:55 +02:00
va_end(args);
}
/* write to stderr, if that's not the logfile already */
if (logfile!=stderr && (log_stderr & mask)) {
int dupe = check_dupe(format, prefix);
if (!dupe) {
va_list args;
va_start(args, format);
_log_writeln(stderr, stdio_codepage, prefix, format, args);
2014-06-17 04:12:55 +02:00
va_end(args);
}
}
if (log_flags & LOG_FLUSH) {
log_flush();
}
}
void log_error(const char *format, ...)
2010-08-08 10:06:34 +02:00
{
2012-05-16 07:21:59 +02:00
const char * prefix = "ERROR";
const int mask = LOG_CPERROR;
2010-08-08 10:06:34 +02:00
2012-05-16 07:21:59 +02:00
/* write to the logfile, always */
if (logfile && (log_flags & mask)) {
va_list args;
va_start(args, format);
_log_writeln(logfile, 0, prefix, format, args);
2012-05-16 07:21:59 +02:00
va_end(args);
2010-08-08 10:06:34 +02:00
}
2012-05-16 07:21:59 +02:00
/* write to stderr, if that's not the logfile already */
if (logfile!=stderr && (log_stderr & mask)) {
int dupe = check_dupe(format, prefix);
if (!dupe) {
va_list args;
va_start(args, format);
_log_writeln(stderr, stdio_codepage, prefix, format, args);
2012-05-16 07:21:59 +02:00
va_end(args);
2010-08-08 10:06:34 +02:00
}
}
2012-05-16 07:21:59 +02:00
if (log_flags & LOG_FLUSH) {
log_flush();
}
2010-08-08 10:06:34 +02:00
}
2011-03-07 08:02:35 +01:00
void log_info(const char *format, ...)
2012-05-16 07:21:59 +02:00
{
const char * prefix = "INFO";
const int mask = LOG_CPINFO;
/* write to the logfile, always */
if (logfile && (log_flags & mask)) {
va_list args;
va_start(args, format);
_log_writeln(logfile, 0, prefix, format, args);
2012-05-16 07:21:59 +02:00
va_end(args);
}
2010-08-08 10:06:34 +02:00
2012-05-16 07:21:59 +02:00
/* write to stderr, if that's not the logfile already */
if (logfile!=stderr && (log_stderr & mask)) {
int dupe = check_dupe(format, prefix);
if (!dupe) {
va_list args;
va_start(args, format);
_log_writeln(stderr, stdio_codepage, prefix, format, args);
2012-05-16 07:21:59 +02:00
va_end(args);
}
}
if (log_flags & LOG_FLUSH) {
log_flush();
}
}
void log_printf(FILE * io, const char *format, ...)
2010-08-08 10:06:34 +02:00
{
2012-05-16 07:21:59 +02:00
const char * prefix = "INFO";
const int mask = LOG_CPINFO;
2011-03-07 08:02:35 +01:00
2012-05-16 07:21:59 +02:00
/* write to the logfile, always */
if (logfile && (log_flags & mask)) {
int codepage = (logfile==stderr || logfile==stdout) ? stdio_codepage : 0;
va_list args;
va_start(args, format);
_log_write(logfile, codepage, prefix, format, args);
va_end(args);
}
/* write to io, if that's not the logfile already */
if (logfile!=io && (log_stderr & mask)) {
int dupe = check_dupe(format, prefix);
if (!dupe) {
va_list args;
va_start(args, format);
_log_write(io, stdio_codepage, prefix, format, args);
va_end(args);
2010-08-08 10:06:34 +02:00
}
}
2012-05-16 07:21:59 +02:00
if (log_flags & LOG_FLUSH) {
log_flush();
}
2010-08-08 10:06:34 +02:00
}