2015-07-06 19:53:09 +02:00
|
|
|
|
/*
|
2010-08-08 10:06:34 +02:00
|
|
|
|
+-------------------+ 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"
|
2015-10-29 09:04:23 +01:00
|
|
|
|
#include "bsdstring.h"
|
2010-08-08 10:06:34 +02:00
|
|
|
|
#include "unicode.h"
|
|
|
|
|
|
2012-06-04 03:41:07 +02:00
|
|
|
|
#include <assert.h>
|
2015-11-04 14:55:13 +01:00
|
|
|
|
#include <errno.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 */
|
2012-05-16 01:56:25 +02:00
|
|
|
|
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 */
|
2012-05-16 09:11:49 +02:00
|
|
|
|
#define LOG_MAXBACKUPS 5
|
2011-03-07 08:02:35 +01:00
|
|
|
|
void log_flush(void)
|
2010-08-08 10:06:34 +02:00
|
|
|
|
{
|
2015-01-30 20:37:14 +01: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
|
|
|
|
{
|
2015-01-30 20:37:14 +01:00
|
|
|
|
fflush(stdout);
|
|
|
|
|
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
|
|
|
|
{
|
2015-01-30 20:37:14 +01:00
|
|
|
|
/* when console output on MSDOS, convert to codepage */
|
|
|
|
|
const char *input = format;
|
|
|
|
|
char *pos = buffer;
|
2010-08-08 10:06:34 +02:00
|
|
|
|
|
2015-01-30 20:37:14 +01:00
|
|
|
|
while (pos + 1 < buffer + length && *input) {
|
2015-07-06 19:53:09 +02:00
|
|
|
|
size_t size = 0;
|
2015-01-30 20:37:14 +01:00
|
|
|
|
int result = 0;
|
|
|
|
|
if (codepage == 437) {
|
2015-07-06 19:53:09 +02:00
|
|
|
|
result = unicode_utf8_to_cp437(pos, input, &size);
|
2015-01-30 20:37:14 +01:00
|
|
|
|
}
|
|
|
|
|
else if (codepage == 1252) {
|
2015-07-06 19:53:09 +02:00
|
|
|
|
result = unicode_utf8_to_cp1252(pos, input, &size);
|
2015-01-30 20:37:14 +01:00
|
|
|
|
}
|
|
|
|
|
if (result != 0) {
|
|
|
|
|
*pos = 0; /* just in case caller ignores our return value */
|
|
|
|
|
return result;
|
|
|
|
|
}
|
|
|
|
|
++pos;
|
|
|
|
|
input += length;
|
2010-08-08 10:06:34 +02:00
|
|
|
|
}
|
2015-01-30 20:37:14 +01:00
|
|
|
|
*pos = 0;
|
|
|
|
|
return 0;
|
2010-08-08 10:06:34 +02:00
|
|
|
|
}
|
|
|
|
|
|
2012-05-16 09:11:49 +02:00
|
|
|
|
void log_rotate(const char *filename, int maxindex)
|
|
|
|
|
{
|
2015-01-30 20:37:14 +01:00
|
|
|
|
if (_access(filename, 4) == 0) {
|
|
|
|
|
char buffer[2][MAX_PATH];
|
2015-11-04 14:16:15 +01:00
|
|
|
|
int dst = 1;
|
2015-01-30 20:37:14 +01:00
|
|
|
|
assert(strlen(filename) < sizeof(buffer[0]) - 4);
|
2015-11-04 14:16:15 +01:00
|
|
|
|
|
|
|
|
|
sprintf(buffer[dst], "%s.%d", filename, maxindex);
|
|
|
|
|
while (maxindex > 0) {
|
|
|
|
|
int err, src = dst - 1;
|
|
|
|
|
sprintf(buffer[src], "%s.%d", filename, --maxindex);
|
|
|
|
|
err = rename(buffer[src], buffer[dst]);
|
|
|
|
|
if (err != 0) {
|
|
|
|
|
log_error("log rotate %s: %s", buffer[dst], strerror(errno));
|
2015-01-30 20:37:14 +01:00
|
|
|
|
}
|
2015-11-04 14:16:15 +01:00
|
|
|
|
dst = src;
|
2015-01-30 20:37:14 +01:00
|
|
|
|
}
|
2015-11-04 14:16:15 +01:00
|
|
|
|
if (rename(filename, buffer[dst]) != 0) {
|
|
|
|
|
log_error("log rotate %s: %s", buffer[dst], strerror(errno));
|
2015-01-30 20:37:14 +01:00
|
|
|
|
}
|
2012-05-16 09:11:49 +02:00
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2011-03-07 08:02:35 +01:00
|
|
|
|
void log_open(const char *filename)
|
2010-08-08 10:06:34 +02:00
|
|
|
|
{
|
2015-01-30 20:37:14 +01:00
|
|
|
|
if (logfile) {
|
|
|
|
|
log_close();
|
|
|
|
|
}
|
|
|
|
|
log_rotate(filename, LOG_MAXBACKUPS);
|
|
|
|
|
logfile = fopen(filename, "a");
|
|
|
|
|
if (logfile) {
|
|
|
|
|
/* Get UNIX-style time and display as number and string. */
|
|
|
|
|
time_t ltime;
|
|
|
|
|
time(<ime);
|
|
|
|
|
fprintf(logfile, "===\n=== Logfile started at %s===\n", ctime(<ime));
|
|
|
|
|
}
|
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
|
|
|
|
{
|
2015-01-30 20:37:14 +01:00
|
|
|
|
if (!logfile || logfile == stderr || logfile == stdout)
|
|
|
|
|
return;
|
|
|
|
|
if (logfile) {
|
|
|
|
|
/* Get UNIX-style time and display as number and string. */
|
|
|
|
|
time_t ltime;
|
|
|
|
|
time(<ime);
|
|
|
|
|
fprintf(logfile, "===\n=== Logfile closed at %s===\n\n", ctime(<ime));
|
|
|
|
|
fclose(logfile);
|
|
|
|
|
}
|
|
|
|
|
logfile = 0;
|
2010-08-08 10:06:34 +02:00
|
|
|
|
}
|
|
|
|
|
|
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
|
|
|
|
{
|
2015-01-30 20:37:14 +01:00
|
|
|
|
static const char *last_type; /* STATIC_XCALL: used across calls */
|
|
|
|
|
static char last_message[32]; /* STATIC_XCALL: used across calls */
|
|
|
|
|
static int dupes = 0; /* STATIC_XCALL: used across calls */
|
|
|
|
|
if (strncmp(last_message, format, sizeof(last_message)) == 0) {
|
|
|
|
|
++dupes;
|
|
|
|
|
return 1;
|
2010-08-08 10:06:34 +02:00
|
|
|
|
}
|
2015-01-30 20:37:14 +01:00
|
|
|
|
if (dupes) {
|
|
|
|
|
if (log_flags & LOG_CPERROR) {
|
|
|
|
|
fprintf(stderr, "%s: last message repeated %d times\n", last_type,
|
|
|
|
|
dupes + 1);
|
|
|
|
|
}
|
|
|
|
|
dupes = 0;
|
|
|
|
|
}
|
2015-10-29 09:04:23 +01:00
|
|
|
|
strlcpy(last_message, format, sizeof(last_message));
|
2015-01-30 20:37:14 +01:00
|
|
|
|
last_type = type;
|
|
|
|
|
return 0;
|
2010-08-08 10:06:34 +02:00
|
|
|
|
}
|
|
|
|
|
|
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 01:56:25 +02:00
|
|
|
|
{
|
2015-01-30 20:37:14 +01:00
|
|
|
|
if (stream) {
|
|
|
|
|
fprintf(stream, "%s: ", prefix);
|
|
|
|
|
if (codepage) {
|
|
|
|
|
char buffer[MAXLENGTH];
|
|
|
|
|
char converted[MAXLENGTH];
|
2012-06-04 03:41:07 +02:00
|
|
|
|
|
2015-01-30 20:37:14 +01:00
|
|
|
|
vsnprintf(buffer, sizeof(buffer), format, args);
|
|
|
|
|
if (cp_convert(buffer, converted, MAXLENGTH, codepage) == 0) {
|
|
|
|
|
fputs(converted, stream);
|
|
|
|
|
}
|
|
|
|
|
else {
|
|
|
|
|
/* fall back to non-converted output */
|
|
|
|
|
vfprintf(stream, format, args);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
else {
|
|
|
|
|
vfprintf(stream, format, args);
|
|
|
|
|
}
|
2012-05-16 01:56:25 +02:00
|
|
|
|
}
|
2012-05-16 07:21:59 +02:00
|
|
|
|
}
|
|
|
|
|
|
2014-08-11 12:02:16 +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);
|
2015-01-30 20:37:14 +01:00
|
|
|
|
if (format[len - 1] != '\n') {
|
2014-08-11 12:02:16 +02:00
|
|
|
|
fputc('\n', stream);
|
|
|
|
|
}
|
|
|
|
|
}
|
2012-05-17 01:56:07 +02:00
|
|
|
|
void log_debug(const char *format, ...)
|
2012-05-16 07:21:59 +02:00
|
|
|
|
{
|
2015-01-30 20:37:14 +01:00
|
|
|
|
const char * prefix = "DEBUG";
|
|
|
|
|
const int mask = LOG_CPDEBUG;
|
2012-05-16 07:21:59 +02:00
|
|
|
|
|
2015-01-30 20:37:14 +01: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);
|
|
|
|
|
va_end(args);
|
|
|
|
|
}
|
2012-05-16 07:21:59 +02:00
|
|
|
|
|
2015-01-30 20:37:14 +01: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);
|
|
|
|
|
va_end(args);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
if (log_flags & LOG_FLUSH) {
|
|
|
|
|
log_flush();
|
2012-05-16 01:56:25 +02:00
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2012-05-17 00:52:37 +02:00
|
|
|
|
void log_warning(const char *format, ...)
|
2010-08-08 10:06:34 +02:00
|
|
|
|
{
|
2015-01-30 20:37:14 +01:00
|
|
|
|
const char * prefix = "WARNING";
|
|
|
|
|
const int mask = LOG_CPWARNING;
|
2010-08-08 10:06:34 +02:00
|
|
|
|
|
2015-01-30 20:37:14 +01: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);
|
|
|
|
|
va_end(args);
|
|
|
|
|
}
|
2014-06-17 04:12:55 +02:00
|
|
|
|
|
2015-01-30 20:37:14 +01: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);
|
|
|
|
|
va_end(args);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
if (log_flags & LOG_FLUSH) {
|
|
|
|
|
log_flush();
|
2014-06-17 04:12:55 +02:00
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2012-05-17 01:52:01 +02:00
|
|
|
|
void log_error(const char *format, ...)
|
2010-08-08 10:06:34 +02:00
|
|
|
|
{
|
2015-01-30 20:37:14 +01:00
|
|
|
|
const char * prefix = "ERROR";
|
|
|
|
|
const int mask = LOG_CPERROR;
|
2010-08-08 10:06:34 +02:00
|
|
|
|
|
2015-01-30 20:37:14 +01: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);
|
|
|
|
|
va_end(args);
|
|
|
|
|
}
|
2010-08-08 10:06:34 +02:00
|
|
|
|
|
2015-01-30 20:37:14 +01: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);
|
|
|
|
|
va_end(args);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
if (log_flags & LOG_FLUSH) {
|
|
|
|
|
log_flush();
|
2010-08-08 10:06:34 +02:00
|
|
|
|
}
|
|
|
|
|
}
|
2011-03-07 08:02:35 +01:00
|
|
|
|
|
2015-02-23 02:47:08 +01:00
|
|
|
|
void log_fatal(const char *format, ...)
|
|
|
|
|
{
|
|
|
|
|
const char * prefix = "ERROR";
|
|
|
|
|
const int mask = LOG_CPERROR;
|
|
|
|
|
|
|
|
|
|
/* write to the logfile, always */
|
|
|
|
|
if (logfile && (log_flags & mask)) {
|
|
|
|
|
va_list args;
|
|
|
|
|
va_start(args, format);
|
|
|
|
|
_log_writeln(logfile, 0, prefix, format, args);
|
|
|
|
|
va_end(args);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/* write to stderr, if that's not the logfile already */
|
|
|
|
|
if (logfile != stderr) {
|
|
|
|
|
int dupe = check_dupe(format, prefix);
|
|
|
|
|
if (!dupe) {
|
|
|
|
|
va_list args;
|
|
|
|
|
va_start(args, format);
|
|
|
|
|
_log_writeln(stderr, stdio_codepage, prefix, format, args);
|
|
|
|
|
va_end(args);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
if (log_flags & LOG_FLUSH) {
|
|
|
|
|
log_flush();
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2012-06-17 07:38:44 +02:00
|
|
|
|
void log_info(const char *format, ...)
|
2012-05-16 07:21:59 +02:00
|
|
|
|
{
|
2015-01-30 20:37:14 +01:00
|
|
|
|
const char * prefix = "INFO";
|
|
|
|
|
const int mask = LOG_CPINFO;
|
2012-05-16 07:21:59 +02:00
|
|
|
|
|
2015-01-30 20:37:14 +01: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);
|
|
|
|
|
va_end(args);
|
|
|
|
|
}
|
2010-08-08 10:06:34 +02:00
|
|
|
|
|
2015-01-30 20:37:14 +01: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);
|
|
|
|
|
va_end(args);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
if (log_flags & LOG_FLUSH) {
|
|
|
|
|
log_flush();
|
2012-05-16 07:21:59 +02:00
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void log_printf(FILE * io, const char *format, ...)
|
2010-08-08 10:06:34 +02:00
|
|
|
|
{
|
2015-01-30 20:37:14 +01:00
|
|
|
|
const char * prefix = "INFO";
|
|
|
|
|
const int mask = LOG_CPINFO;
|
2011-03-07 08:02:35 +01:00
|
|
|
|
|
2015-01-30 20:37:14 +01: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);
|
|
|
|
|
}
|
2012-05-16 07:21:59 +02:00
|
|
|
|
|
2015-01-30 20:37:14 +01:00
|
|
|
|
/* 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);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
if (log_flags & LOG_FLUSH) {
|
|
|
|
|
log_flush();
|
2010-08-08 10:06:34 +02:00
|
|
|
|
}
|
|
|
|
|
}
|