From 729c4ceea12d432f7a7a884d3f2eda4b46a54d18 Mon Sep 17 00:00:00 2001 From: Enno Rehling Date: Thu, 28 Jan 2016 16:00:36 +0100 Subject: [PATCH 01/15] increase error-logging from unit tests. suppress some unrelated errors. --- src/kernel/messages.c | 2 +- src/laws.test.c | 2 ++ src/spy.test.c | 6 +++--- src/study.test.c | 3 ++- src/test_eressea.c | 2 +- src/tests.c | 15 +++++++++++++-- 6 files changed, 22 insertions(+), 8 deletions(-) diff --git a/src/kernel/messages.c b/src/kernel/messages.c index 779354498..330c2feeb 100644 --- a/src/kernel/messages.c +++ b/src/kernel/messages.c @@ -90,7 +90,7 @@ struct message *msg_feedback(const struct unit *u, struct order *ord, ord = u->thisorder; if (!mtype) { - log_error("trying to create message of unknown type \"%s\"\n", name); + log_warning("trying to create message of unknown type \"%s\"\n", name); if (!mt_find("missing_feedback")) { mt_register(mt_new_va("missing_feedback", "unit:unit", "region:region", "command:order", "name:string", 0)); } diff --git a/src/laws.test.c b/src/laws.test.c index 5d21250cd..5be4c7143 100644 --- a/src/laws.test.c +++ b/src/laws.test.c @@ -1229,6 +1229,8 @@ static void test_show_without_item(CuTest *tc) test_cleanup(); loc = get_or_create_locale("de"); + locale_setstring(loc, parameters[P_ANY], "ALLE"); + init_parameters(loc); r = test_create_region(0, 0, test_create_terrain("testregion", LAND_REGION)); f = test_create_faction(test_create_race("human")); diff --git a/src/spy.test.c b/src/spy.test.c index 00a1767d0..a4e2f1151 100644 --- a/src/spy.test.c +++ b/src/spy.test.c @@ -100,7 +100,7 @@ static void test_sabotage_self(CuTest *tc) { order *ord; setup_sabotage(); - r = test_create_region(0, 0, NULL); + r = findregion(0, 0); assert(r); u = test_create_unit(test_create_faction(NULL), r); assert(u && u->faction && u->region == r); @@ -122,7 +122,7 @@ static void test_sabotage_other_fail(CuTest *tc) { message *msg; setup_sabotage(); - r = test_create_region(0, 0, NULL); + r = findregion(0, 0); assert(r); u = test_create_unit(test_create_faction(NULL), r); u2 = test_create_unit(test_create_faction(NULL), r); @@ -151,7 +151,7 @@ static void test_sabotage_other_success(CuTest *tc) { order *ord; setup_sabotage(); - r = test_create_region(0, 0, NULL); + r = findregion(0, 0); assert(r); u = test_create_unit(test_create_faction(NULL), r); u2 = test_create_unit(test_create_faction(NULL), r); diff --git a/src/study.test.c b/src/study.test.c index 6b9d97787..bdbfb9a01 100644 --- a/src/study.test.c +++ b/src/study.test.c @@ -6,6 +6,7 @@ #include #include #include +#include #include #include #include @@ -29,7 +30,7 @@ static void setup_study(study_fixture *fix, skill_t sk) { test_cleanup(); config_set("study.random_progress", "0"); test_create_world(); - r = test_create_region(0, 0, 0); + r = findregion(0, 0); f = test_create_faction(0); lang = get_or_create_locale(locale_name(f->locale)); locale_setstring(lang, mkname("skill", skillnames[sk]), skillnames[sk]); diff --git a/src/test_eressea.c b/src/test_eressea.c index 1053306c6..9e5fba524 100644 --- a/src/test_eressea.c +++ b/src/test_eressea.c @@ -156,7 +156,7 @@ int RunAllTests(int argc, char *argv[]) } int main(int argc, char ** argv) { - log_stderr = 0; + log_stderr = LOG_CPERROR|LOG_FLUSH; ++argv; --argc; if (argc > 0 && strcmp("--list", argv[0]) == 0) { diff --git a/src/tests.c b/src/tests.c index 6fef425d0..425e01d43 100644 --- a/src/tests.c +++ b/src/tests.c @@ -45,14 +45,19 @@ struct race *test_create_race(const char *name) struct region *test_create_region(int x, int y, const terrain_type *terrain) { - region *r = new_region(x, y, NULL, 0); + region *r = findregion(x, y); + if (!r) { + r = new_region(x, y, NULL, 0); + } if (!terrain) { terrain_type *t = get_or_create_terrain("plain"); t->size = 1000; fset(t, LAND_REGION|CAVALRY_REGION|FOREST_REGION); terraform_region(r, t); - } else + } + else { terraform_region(r, terrain); + } rsettrees(r, 0, 0); rsettrees(r, 1, 0); rsettrees(r, 2, 0); @@ -255,6 +260,11 @@ void test_create_world(void) struct locale * loc; loc = get_or_create_locale("de"); + + locale_setstring(loc, parameters[P_SHIP], "SCHIFF"); + locale_setstring(loc, parameters[P_ANY], "ALLE"); + init_parameters(loc); + locale_setstring(loc, keyword(K_RESERVE), "RESERVIEREN"); locale_setstring(loc, "money", "SILBER"); init_resources(); @@ -270,6 +280,7 @@ void test_create_world(void) test_create_itemtype("iron"); test_create_itemtype("stone"); + t_plain = test_create_terrain("plain", LAND_REGION | FOREST_REGION | WALK_INTO | CAVALRY_REGION | SAIL_INTO | FLY_INTO); t_plain->size = 1000; t_plain->max_road = 100; From 9bf1059d8a5364568dbca46e3671da8fcaf87917 Mon Sep 17 00:00:00 2001 From: Enno Rehling Date: Fri, 29 Jan 2016 17:49:27 +0100 Subject: [PATCH 02/15] trying to make tests not throw out so many ERROR log mesages, commenting on some of them. --- src/kernel/config.c | 46 +++++++++++++++++++++++++---------------- src/kernel/save.test.c | 1 + src/kernel/spell.test.c | 3 +++ src/magic.test.c | 3 +++ src/reports.test.c | 1 + src/tests.c | 2 +- 6 files changed, 37 insertions(+), 19 deletions(-) diff --git a/src/kernel/config.c b/src/kernel/config.c index bd9a30d81..21907ecc3 100644 --- a/src/kernel/config.c +++ b/src/kernel/config.c @@ -640,9 +640,32 @@ int check_param(const struct param *p, const char *key, const char *searchvalue) return result; } -const char * relpath(char *buf, size_t sz, const char *path) { - strlcpy(buf, basepath(), sz); - strlcat(buf, path, sz); +static const char *g_basedir; +const char *basepath(void) +{ + if (g_basedir) + return g_basedir; + return "."; +} + +void set_basepath(const char *path) +{ + g_basedir = path; +} + +static const char * relpath(char *buf, size_t sz, const char *path) { + if (g_basedir) { + strlcpy(buf, g_basedir, sz); +#ifdef WIN32 + strlcat(buf, "\\", sz); +#else + strlcat(buf, "/", sz); +#endif + strlcat(buf, path, sz); + } + else { + strlcpy(buf, path, sz); + } return buf; } @@ -652,7 +675,7 @@ const char *datapath(void) static char zText[MAX_PATH]; // FIXME: static return value if (g_datadir) return g_datadir; - return relpath(zText, sizeof(zText), "/data"); + return relpath(zText, sizeof(zText), "data"); } void set_datapath(const char *path) @@ -666,7 +689,7 @@ const char *reportpath(void) static char zText[MAX_PATH]; // FIXME: static return value if (g_reportdir) return g_reportdir; - return relpath(zText, sizeof(zText), "/reports"); + return relpath(zText, sizeof(zText), "reports"); } void set_reportpath(const char *path) @@ -674,19 +697,6 @@ void set_reportpath(const char *path) g_reportdir = path; } -static const char *g_basedir; -const char *basepath(void) -{ - if (g_basedir) - return g_basedir; - return "."; -} - -void set_basepath(const char *path) -{ - g_basedir = path; -} - double get_param_flt(const struct param *p, const char *key, double def) { const char *str = get_param(p, key); diff --git a/src/kernel/save.test.c b/src/kernel/save.test.c index 35e07325b..5395344a2 100644 --- a/src/kernel/save.test.c +++ b/src/kernel/save.test.c @@ -32,6 +32,7 @@ static void test_readwrite_unit(CuTest * tc) struct region *r; struct faction *f; int fno; + /* FIXME: at some point during this test, errno is set to 17 (File exists), why? */ test_cleanup(); r = test_create_region(0, 0, 0); diff --git a/src/kernel/spell.test.c b/src/kernel/spell.test.c index 23e227ebe..0c0beb6ce 100644 --- a/src/kernel/spell.test.c +++ b/src/kernel/spell.test.c @@ -2,6 +2,7 @@ #include #include +#include #include #include @@ -24,6 +25,7 @@ static void test_create_a_spell(CuTest * tc) static void test_create_duplicate_spell(CuTest * tc) { spell *sp; + /* FIXME: this test emits ERROR messages (duplicate spells), inject a logger to verify that */ test_cleanup(); CuAssertPtrEquals(tc, 0, find_spell("testspell")); @@ -36,6 +38,7 @@ static void test_create_duplicate_spell(CuTest * tc) static void test_create_spell_with_id(CuTest * tc) { spell *sp; + /* FIXME: this test emits ERROR messages (duplicate spells), inject a logger to verify that */ test_cleanup(); CuAssertPtrEquals(tc, 0, find_spellbyid(42)); diff --git a/src/magic.test.c b/src/magic.test.c index 6ed4d81a9..c531f4be1 100644 --- a/src/magic.test.c +++ b/src/magic.test.c @@ -2,6 +2,7 @@ #include "magic.h" +#include #include #include #include @@ -386,6 +387,8 @@ void test_multi_cast(CuTest *tc) { u = test_create_unit(test_create_faction(0), test_create_region(0, 0, 0)); u->faction->locale = lang = get_or_create_locale("de"); + locale_setstring(lang, parameters[P_ANY], "ALLE"); + init_parameters(lang); locale_setstring(lang, mkname("spell", sp->sname), "Feuerball"); CuAssertStrEquals(tc, "Feuerball", spell_name(sp, lang)); set_level(u, SK_MAGIC, 10); diff --git a/src/reports.test.c b/src/reports.test.c index 0955e8a6f..103dc28ff 100644 --- a/src/reports.test.c +++ b/src/reports.test.c @@ -253,6 +253,7 @@ static void test_write_unit(CuTest *tc) { race *rc; struct locale *lang; char buffer[1024]; + /* FIXME: test emits ERROR: no translation for combat status status_aggressive in locale de */ test_cleanup(); rc = rc_get_or_create("human"); diff --git a/src/tests.c b/src/tests.c index 425e01d43..0511919c3 100644 --- a/src/tests.c +++ b/src/tests.c @@ -108,7 +108,7 @@ void test_cleanup(void) if (errno) { int error = errno; errno = 0; - log_error("errno: %d", error); + log_error("errno: %d (%s)", error, strerror(error)); } random_source_reset(); From 73f16d5bb61cf2ac16a24c61a1d9ffc122c2811e Mon Sep 17 00:00:00 2001 From: Enno Rehling Date: Fri, 29 Jan 2016 19:11:48 +0100 Subject: [PATCH 03/15] adding a join_path function to try and narrow down misbehavior. --- src/kernel/config.c | 25 ++++++++++++++++++------- src/kernel/config.h | 1 + src/kernel/save.test.c | 2 +- 3 files changed, 20 insertions(+), 8 deletions(-) diff --git a/src/kernel/config.c b/src/kernel/config.c index 21907ecc3..6cc31c3b0 100644 --- a/src/kernel/config.c +++ b/src/kernel/config.c @@ -653,15 +653,26 @@ void set_basepath(const char *path) g_basedir = path; } +#ifdef WIN32 +#define PATH_DELIM '\\' +#else +#define PATH_DELIM '/' +#endif + + +char * join_path(const char *p1, const char *p2, char *dst, size_t len) { + size_t sz; + assert(p1 && p2); + sz = strlcpy(dst, p1, len); + assert(sz < len); + dst[sz++] = PATH_DELIM; + strlcpy(dst + sz, p2, len - sz); + return dst; +} + static const char * relpath(char *buf, size_t sz, const char *path) { if (g_basedir) { - strlcpy(buf, g_basedir, sz); -#ifdef WIN32 - strlcat(buf, "\\", sz); -#else - strlcat(buf, "/", sz); -#endif - strlcat(buf, path, sz); + join_path(g_basedir, path, buf, sz); } else { strlcpy(buf, path, sz); diff --git a/src/kernel/config.h b/src/kernel/config.h index 257de940d..30aa25549 100644 --- a/src/kernel/config.h +++ b/src/kernel/config.h @@ -185,6 +185,7 @@ struct param; double config_get_flt(const char *key, double def); bool config_token(const char *key, const char *tok); + char * join_path(const char *p1, const char *p2, char *dst, size_t len); bool ExpensiveMigrants(void); int NMRTimeout(void); int LongHunger(const struct unit *u); diff --git a/src/kernel/save.test.c b/src/kernel/save.test.c index 5395344a2..1492750ab 100644 --- a/src/kernel/save.test.c +++ b/src/kernel/save.test.c @@ -40,7 +40,7 @@ static void test_readwrite_unit(CuTest * tc) fno = f->no; u = test_create_unit(f, r); _mkdir(datapath()); - sprintf(path, "%s/%s", datapath(), filename); + join_path(datapath(), filename, path, sizeof(path)); data = gamedata_open(path, "wb"); CuAssertPtrNotNull(tc, data); // TODO: intermittent test From 8f4e6475c7d1c3cac84f02bdc52e2dec7a0954df Mon Sep 17 00:00:00 2001 From: Enno Rehling Date: Sun, 31 Jan 2016 19:54:49 +0100 Subject: [PATCH 04/15] complete redesign of the log library (almost finished?) --- src/main.c | 6 +- src/test_eressea.c | 4 +- src/util/log.c | 419 ++++++++++++++++++--------------------------- src/util/log.h | 15 +- 4 files changed, 189 insertions(+), 255 deletions(-) diff --git a/src/main.c b/src/main.c index 6eac28470..a19c2c09e 100644 --- a/src/main.c +++ b/src/main.c @@ -130,7 +130,7 @@ static int get_arg(int argc, char **argv, size_t len, int index, const char **re static int parse_args(int argc, char **argv, int *exitcode) { - int i; + int i, log_stderr = 0; for (i = 1; i != argc; ++i) { char *argi = argv[i]; @@ -210,6 +210,10 @@ static int parse_args(int argc, char **argv, int *exitcode) break; } + if (log_stderr) { + log_to_file(log_stderr, stderr); + } + return 0; } diff --git a/src/test_eressea.c b/src/test_eressea.c index 9e5fba524..616ed63ef 100644 --- a/src/test_eressea.c +++ b/src/test_eressea.c @@ -5,6 +5,8 @@ #include #include #include +#include +#include #include #pragma warning(disable: 4210) @@ -156,7 +158,7 @@ int RunAllTests(int argc, char *argv[]) } int main(int argc, char ** argv) { - log_stderr = LOG_CPERROR|LOG_FLUSH; + log_to_file(LOG_CPERROR, stderr); ++argv; --argc; if (argc > 0 && strcmp("--list", argv[0]) == 0) { diff --git a/src/util/log.c b/src/util/log.c index 3b7ec50a0..1cc3836db 100644 --- a/src/util/log.c +++ b/src/util/log.c @@ -24,29 +24,33 @@ without prior permission by the authors of Eressea. /* TODO: set from external function */ int log_flags = LOG_FLUSH | LOG_CPERROR | LOG_CPWARNING | LOG_CPDEBUG; -int log_stderr = LOG_FLUSH | LOG_CPERROR | LOG_CPWARNING; #ifdef STDIO_CP static int stdio_codepage = STDIO_CP; #else static int stdio_codepage = 0; #endif -static FILE *logfile; + +typedef struct logger { + void(*log)(void *data, int level, const char *module, const char *format, va_list args); + void *data; + int flags; + struct logger *next; +} logger; + +static logger *loggers; + +void log_create(int flags, void *data, log_fun call) { + logger *lgr = malloc(sizeof(logger)); + lgr->log = call; + lgr->flags = flags; + lgr->data = data; + lgr->next = loggers; + loggers = lgr; +} #define MAXLENGTH 4096 /* because I am lazy, CP437 output is limited to this many chars */ #define LOG_MAXBACKUPS 5 -void log_flush(void) -{ - if (logfile) fflush(logfile); -} - -void log_puts(const char *str) -{ - fflush(stdout); - if (logfile) { - fputs(str, logfile); - } -} static int cp_convert(const char *format, char *buffer, size_t length, int codepage) @@ -100,6 +104,157 @@ void log_rotate(const char *filename, int maxindex) } } +static const char *log_prefix(int level) { + const char * prefix = "ERROR"; + if (level == LOG_CPWARNING) prefix = "WARNING"; + else if (level == LOG_CPDEBUG) prefix = "DEBUG"; + else if (level == LOG_CPINFO) prefix = "INFO"; + return prefix; +} + +static int check_dupe(const char *format, int type) +{ + static int 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; + } + if (dupes) { + if (log_flags & LOG_CPERROR) { + fprintf(stderr, "%s: last message repeated %d times\n", log_prefix(last_type), + dupes + 1); + } + dupes = 0; + } + strlcpy(last_message, format, sizeof(last_message)); + last_type = type; + return 0; +} + +static void _log_write(FILE * stream, int codepage, const char *format, va_list args) +{ + if (codepage) { + char buffer[MAXLENGTH]; + char converted[MAXLENGTH]; + + 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); + } +} + +static void log_stdio(void *data, int level, const char *module, const char *format, va_list args) { + FILE *out = (FILE *)data; + int codepage = (out == stderr || out == stdout) ? stdio_codepage : 0; + const char *prefix = log_prefix(level); + size_t len = strlen(format); + + fprintf(out, "%s: ", prefix); + + _log_write(out, codepage, format, args); + if (format[len - 1] != '\n') { + fputc('\n', out); + } +} + +void log_to_file(int flags, FILE *out) { + log_create(flags, out, log_stdio); +} + +static void log_write(int flags, const char *module, const char *format, va_list args) { + logger *lg; + for (lg = loggers; lg; lg = lg->next) { + int level = flags & LOG_LEVELS; + if (lg->flags & level) { + int dupe = 0; + if (lg->flags & LOG_BRIEF) { + dupe = check_dupe(format, level); + } + if (dupe == 0) { + lg->log(lg->data, level, NULL, format, args); + } + } + } +} + + +void log_fatal(const char *format, ...) +{ + va_list args; + va_start(args, format); + log_write(LOG_CPERROR, NULL, format, args); + va_end(args); +} + +void log_error(const char *format, ...) +{ + va_list args; + va_start(args, format); + log_write(LOG_CPERROR, NULL, format, args); + va_end(args); +} + +void log_warning(const char *format, ...) +{ + va_list args; + va_start(args, format); + log_write(LOG_CPWARNING, NULL, format, args); + va_end(args); +} + +void log_debug(const char *format, ...) +{ + va_list args; + va_start(args, format); + log_write(LOG_CPDEBUG, NULL, format, args); + va_end(args); +} + +void log_info(const char *format, ...) +{ + va_list args; + va_start(args, format); + log_write(LOG_CPINFO, NULL, format, args); + va_end(args); +} + +void log_printf(FILE * io, const char *format, ...) +{ + va_list args; + va_start(args, format); + log_write(LOG_CPINFO, NULL, format, args); + va_end(args); +} + + +static FILE *logfile; + +void log_close(void) +{ + while (loggers) { + logger *lgr = loggers; + loggers = lgr->next; + free(lgr); + } + if (logfile) { + time_t ltime; + time(<ime); + fprintf(logfile, "===\n=== Logfile closed at %s===\n\n", ctime(<ime)); + fclose(logfile); + } + logfile = 0; +} + void log_open(const char *filename) { if (logfile) { @@ -112,240 +267,6 @@ void log_open(const char *filename) time_t ltime; time(<ime); fprintf(logfile, "===\n=== Logfile started at %s===\n", ctime(<ime)); - } -} - -void log_close(void) -{ - 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; -} - -static int check_dupe(const char *format, const char *type) -{ - 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; - } - if (dupes) { - if (log_flags & LOG_CPERROR) { - fprintf(stderr, "%s: last message repeated %d times\n", last_type, - dupes + 1); - } - dupes = 0; - } - strlcpy(last_message, format, sizeof(last_message)); - last_type = type; - return 0; -} - -static void _log_write(FILE * stream, int codepage, const char * prefix, const char *format, va_list args) -{ - if (stream) { - fprintf(stream, "%s: ", prefix); - if (codepage) { - char buffer[MAXLENGTH]; - char converted[MAXLENGTH]; - - 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); - } - } -} - -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, ...) -{ - 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); - 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); - va_end(args); - } - } - if (log_flags & LOG_FLUSH) { - log_flush(); - } -} - -void log_warning(const char *format, ...) -{ - const char * prefix = "WARNING"; - const int mask = LOG_CPWARNING; - - /* 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 && (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(); - } -} - -void log_error(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 && (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(); - } -} - -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(); - } -} - -void log_info(const char *format, ...) -{ - 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); - 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); - va_end(args); - } - } - if (log_flags & LOG_FLUSH) { - log_flush(); - } -} - -void log_printf(FILE * io, const char *format, ...) -{ - const char * prefix = "INFO"; - const int mask = LOG_CPINFO; - - /* 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); - } - } - if (log_flags & LOG_FLUSH) { - log_flush(); + log_create(log_flags, logfile, log_stdio); } } diff --git a/src/util/log.h b/src/util/log.h index 40a667de4..160951afe 100644 --- a/src/util/log.h +++ b/src/util/log.h @@ -27,11 +27,18 @@ extern "C" { extern void log_info(const char *format, ...); extern void log_printf(FILE * ios, const char *format, ...); -#define LOG_FLUSH 0x01 +#define LOG_CPERROR 0x01 #define LOG_CPWARNING 0x02 -#define LOG_CPERROR 0x04 -#define LOG_CPINFO 0x08 -#define LOG_CPDEBUG 0x10 +#define LOG_CPINFO 0x04 +#define LOG_CPDEBUG 0x08 +#define LOG_LEVELS 0x0F +#define LOG_FLUSH 0x10 +#define LOG_BRIEF 0x20 + + typedef void(*log_fun)(void *data, int level, const char *module, const char *format, va_list args); + + void log_create(int flags, void *data, log_fun call); + void log_to_file(int flags, FILE *out); extern int log_flags; extern int log_stderr; From 3e57b19d6232f70ce5fee2b771c753889fb30f4d Mon Sep 17 00:00:00 2001 From: Enno Rehling Date: Sun, 31 Jan 2016 21:36:01 +0100 Subject: [PATCH 05/15] stop duplicate parsing of arguments --- src/main.c | 4 ---- src/util/log.c | 4 ---- 2 files changed, 8 deletions(-) diff --git a/src/main.c b/src/main.c index a19c2c09e..efcbc6203 100644 --- a/src/main.c +++ b/src/main.c @@ -288,10 +288,6 @@ int main(int argc, char **argv) int err = 0; lua_State *L; setup_signal_handler(); - /* parse args once to read config file location */ - if (parse_args(argc, argv, &err) != 0) { - return err; - } /* ini file sets defaults for arguments*/ parse_config(inifile); if (!global.inifile) { diff --git a/src/util/log.c b/src/util/log.c index 1cc3836db..958e26e2f 100644 --- a/src/util/log.c +++ b/src/util/log.c @@ -236,7 +236,6 @@ void log_printf(FILE * io, const char *format, ...) va_end(args); } - static FILE *logfile; void log_close(void) @@ -257,9 +256,6 @@ void log_close(void) void log_open(const char *filename) { - if (logfile) { - log_close(); - } log_rotate(filename, LOG_MAXBACKUPS); logfile = fopen(filename, "a"); if (logfile) { From 217fbca65624fed3c87898aa39a92fed357f7955 Mon Sep 17 00:00:00 2001 From: Enno Rehling Date: Sun, 31 Jan 2016 21:39:28 +0100 Subject: [PATCH 06/15] missing include broke the gcc build --- src/util/log.h | 3 ++- 1 file changed, 2 insertions(+), 1 deletion(-) diff --git a/src/util/log.h b/src/util/log.h index 160951afe..8fd8f2a95 100644 --- a/src/util/log.h +++ b/src/util/log.h @@ -15,9 +15,10 @@ without prior permission by the authors of Eressea. extern "C" { #endif #include +#include + extern void log_open(const char *filename); extern void log_close(void); - extern void log_flush(void); /* use macros above instead of these: */ extern void log_fatal(const char *format, ...); From b0bb566f47f9c7f1f34b57ce50ac95e798db58e8 Mon Sep 17 00:00:00 2001 From: Enno Rehling Date: Sun, 31 Jan 2016 21:40:19 +0100 Subject: [PATCH 07/15] clang is picky about the correct name of stdarg.h --- src/test_eressea.c | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/src/test_eressea.c b/src/test_eressea.c index 616ed63ef..0a9de8de4 100644 --- a/src/test_eressea.c +++ b/src/test_eressea.c @@ -6,7 +6,7 @@ #include #include #include -#include +#include #include #pragma warning(disable: 4210) From caac2e65b08292f57fc394cb9e77b5e9ed2942d2 Mon Sep 17 00:00:00 2001 From: Enno Rehling Date: Sun, 31 Jan 2016 21:41:25 +0100 Subject: [PATCH 08/15] remove unused includes --- src/test_eressea.c | 2 -- 1 file changed, 2 deletions(-) diff --git a/src/test_eressea.c b/src/test_eressea.c index 0a9de8de4..a9b9b941c 100644 --- a/src/test_eressea.c +++ b/src/test_eressea.c @@ -5,8 +5,6 @@ #include #include #include -#include -#include #include #pragma warning(disable: 4210) From 7ebfdb07475cd15b9d7a461f26f6f57cdcd4b335 Mon Sep 17 00:00:00 2001 From: Enno Rehling Date: Sun, 31 Jan 2016 22:10:37 +0100 Subject: [PATCH 09/15] manually smooth out a merge of two commits trying to do the same thing. prevent negative parents https://bugs.eressea.de/view.php?id=2183 --- src/laws.c | 26 ++++++++++++-------------- 1 file changed, 12 insertions(+), 14 deletions(-) diff --git a/src/laws.c b/src/laws.c index 8b2f781c3..d0239e862 100755 --- a/src/laws.c +++ b/src/laws.c @@ -710,21 +710,19 @@ void immigration(void) } /* Genereate some (0-6 depending on the income) peasants out of nothing */ /* if less than 50 are in the region and there is space and no monster or demon units in the region */ - int peasants = rpeasants(r); - int income = wage(r, NULL, NULL, turn) - maintenance_cost(NULL); - if (repopulate && r->land && (peasants < repopulate) && maxworkingpeasants(r) > (peasants + 30) * 2 && income >= 0) { - int badunit = 0; - unit *u; - for (u = r->units; u; u = u->next) { - if (!playerrace(u_race(u)) || u_race(u) == get_race(RC_DAEMON)) { - badunit = 1; - break; + if (repopulate) { + int peasants = rpeasants(r); + int income = wage(r, NULL, NULL, turn) - maintenance_cost(NULL); + if (income >= 0 && r->land && (peasants < repopulate) && maxworkingpeasants(r) >(peasants + 30) * 2) { + int badunit = 0; + unit *u; + for (u = r->units; u; u = u->next) { + if (!playerrace(u_race(u)) || u_race(u) == get_race(RC_DAEMON)) { + badunit = 1; + break; + } } - } - if (badunit == 0) - { - int x = wage(r, NULL, NULL, turn) - 9; - if (x>0) { + if (badunit == 0) { peasants += (int)(rng_double()*x); rsetpeasants(r, peasants); } From ca81dadedbc0ac1ac17f971d11d8b44f58829ff0 Mon Sep 17 00:00:00 2001 From: Enno Rehling Date: Sun, 31 Jan 2016 22:12:08 +0100 Subject: [PATCH 10/15] missed a spot! --- src/laws.c | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/src/laws.c b/src/laws.c index d0239e862..c3be6f136 100755 --- a/src/laws.c +++ b/src/laws.c @@ -723,7 +723,7 @@ void immigration(void) } } if (badunit == 0) { - peasants += (int)(rng_double()*x); + peasants += (int)(rng_double()*income); rsetpeasants(r, peasants); } } From ba39fc9284357d3c6d20df17875321f0d23a8629 Mon Sep 17 00:00:00 2001 From: Enno Rehling Date: Sun, 31 Jan 2016 22:17:55 +0100 Subject: [PATCH 11/15] I'm really bad at reading. Fixed! --- src/laws.c | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/src/laws.c b/src/laws.c index c3be6f136..eb079b8ac 100755 --- a/src/laws.c +++ b/src/laws.c @@ -712,7 +712,7 @@ void immigration(void) /* if less than 50 are in the region and there is space and no monster or demon units in the region */ if (repopulate) { int peasants = rpeasants(r); - int income = wage(r, NULL, NULL, turn) - maintenance_cost(NULL); + int income = wage(r, NULL, NULL, turn) - maintenance_cost(NULL) + 1; if (income >= 0 && r->land && (peasants < repopulate) && maxworkingpeasants(r) >(peasants + 30) * 2) { int badunit = 0; unit *u; From 2f2bbb16abf45db53cdfb377ab2c33428439c912 Mon Sep 17 00:00:00 2001 From: Enno Rehling Date: Mon, 1 Feb 2016 09:26:24 +0100 Subject: [PATCH 12/15] remove wild mkdir calls all over the code, catch EEXIST errors use join_path more consistently fix a test that's intermittent on windows because fopen(..., "wb") fails, why? --- src/kernel/config.c | 14 ++++++++++++++ src/kernel/config.h | 2 ++ src/kernel/save.c | 13 ++++--------- src/kernel/save.test.c | 4 ++-- src/main.c | 2 +- src/reports.c | 23 ++--------------------- 6 files changed, 25 insertions(+), 33 deletions(-) diff --git a/src/kernel/config.c b/src/kernel/config.c index 6cc31c3b0..08b7f53c9 100644 --- a/src/kernel/config.c +++ b/src/kernel/config.c @@ -708,6 +708,20 @@ void set_reportpath(const char *path) g_reportdir = path; } +int create_directories(void) { + int err; + err = _mkdir(datapath()); + if (err) { + if (errno == EEXIST) errno = 0; + else return err; + } + err = _mkdir(reportpath()); + if (err && errno == EEXIST) { + errno = 0; + } + return err; +} + double get_param_flt(const struct param *p, const char *key, double def) { const char *str = get_param(p, key); diff --git a/src/kernel/config.h b/src/kernel/config.h index 30aa25549..dda7873aa 100644 --- a/src/kernel/config.h +++ b/src/kernel/config.h @@ -148,6 +148,8 @@ struct param; const char *reportpath(void); void set_reportpath(const char *); + int create_directories(void); + void kernel_init(void); void kernel_done(void); diff --git a/src/kernel/save.c b/src/kernel/save.c index 2d37c6822..a4227b9ac 100644 --- a/src/kernel/save.c +++ b/src/kernel/save.c @@ -1747,21 +1747,16 @@ int writegame(const char *filename) stream strm; FILE *F; - sprintf(path, "%s/%s", datapath(), filename); + create_directories(); + join_path(datapath(), filename, path, sizeof(path)); #ifdef HAVE_UNISTD_H /* make sure we don't overwrite an existing file (hard links) */ unlink(path); #endif F = fopen(path, "wb"); if (!F) { - /* we might be missing the directory, let's try creating it */ - int err = _mkdir(datapath()); - if (err) return err; - F = fopen(path, "wb"); - if (!F) { - perror(path); - return -1; - } + perror(path); + return -1; } gdata.store = &store; diff --git a/src/kernel/save.test.c b/src/kernel/save.test.c index 1492750ab..9e9ba9107 100644 --- a/src/kernel/save.test.c +++ b/src/kernel/save.test.c @@ -34,15 +34,15 @@ static void test_readwrite_unit(CuTest * tc) int fno; /* FIXME: at some point during this test, errno is set to 17 (File exists), why? */ + create_directories(); test_cleanup(); r = test_create_region(0, 0, 0); f = test_create_faction(0); fno = f->no; u = test_create_unit(f, r); - _mkdir(datapath()); join_path(datapath(), filename, path, sizeof(path)); - data = gamedata_open(path, "wb"); + data = gamedata_open(path, "w"); CuAssertPtrNotNull(tc, data); // TODO: intermittent test write_unit(data, u); gamedata_close(data); diff --git a/src/main.c b/src/main.c index efcbc6203..305d4a901 100644 --- a/src/main.c +++ b/src/main.c @@ -211,7 +211,7 @@ static int parse_args(int argc, char **argv, int *exitcode) } if (log_stderr) { - log_to_file(log_stderr, stderr); + log_to_file(log_stderr | LOG_FLUSH | LOG_BRIEF, stderr); } return 0; diff --git a/src/reports.c b/src/reports.c index 99b9f755c..b33f5f1cb 100644 --- a/src/reports.c +++ b/src/reports.c @@ -1540,16 +1540,6 @@ static void prepare_report(struct report_context *ctx, faction *f) ctx->last = lastregion(f); } -static void mkreportdir(const char *rpath) { - if (_mkdir(rpath) != 0) { - if (_access(rpath, 0) < 0) { - log_error("could not create reports directory %s: %s", rpath, strerror(errno)); - abort(); - } - } - errno = 0; -} - int write_reports(faction * f, time_t ltime) { unsigned int backup = 1, maxbackup = 128 * 1000; @@ -1557,14 +1547,12 @@ int write_reports(faction * f, time_t ltime) struct report_context ctx; const char *encoding = "UTF-8"; report_type *rtype; - const char *path = reportpath(); if (noreports) { return false; } prepare_report(&ctx, f); get_addresses(&ctx); - mkreportdir(path); // FIXME: too many mkdir calls! init_reports is enough log_debug("Reports for %s:", factionname(f)); for (rtype = report_types; rtype != NULL; rtype = rtype->next) { if (f->options & rtype->flag) { @@ -1635,14 +1623,8 @@ static void check_messages_exist(void) { int init_reports(void) { check_messages_exist(); - + create_directories(); prepare_reports(); - { - if (_access(reportpath(), 0) != 0) { - return 0; - } - } - mkreportdir(reportpath()); return 0; } @@ -1659,8 +1641,7 @@ int reports(void) report_donations(); remove_empty_units(); - mkreportdir(rpath); // FIXME: init_reports already does this? - sprintf(path, "%s/reports.txt", rpath); + join_path(rpath, "reports.txt", path, sizeof(path)); mailit = fopen(path, "w"); if (mailit == NULL) { log_error("%s could not be opened!\n", path); From 85010e53582cb2d14fc81c33c9eb7f89224175ac Mon Sep 17 00:00:00 2001 From: Enno Rehling Date: Mon, 1 Feb 2016 12:27:13 +0100 Subject: [PATCH 13/15] seems like there is a problem opening files with the optional 'b' mode, so remove it everywhere? --- src/bind_storage.c | 2 +- src/kernel/group.test.c | 4 ++-- src/kernel/save.c | 6 +++--- src/kernel/save.test.c | 2 +- storage | 2 +- 5 files changed, 8 insertions(+), 8 deletions(-) diff --git a/src/bind_storage.c b/src/bind_storage.c index ca83f199a..aaeb84128 100644 --- a/src/bind_storage.c +++ b/src/bind_storage.c @@ -32,7 +32,7 @@ without prior permission by the authors of Eressea. static int tolua_storage_create(lua_State * L) { const char *filename = tolua_tostring(L, 1, 0); - const char *type = tolua_tostring(L, 2, "rb"); + const char *type = tolua_tostring(L, 2, "r"); gamedata *data; data = gamedata_open(filename, type); diff --git a/src/kernel/group.test.c b/src/kernel/group.test.c index 46d794d2d..35b25aad5 100644 --- a/src/kernel/group.test.c +++ b/src/kernel/group.test.c @@ -23,7 +23,7 @@ static void test_group_readwrite(CuTest * tc) FILE *F; stream strm; - F = fopen("test.dat", "wb"); + F = fopen("test.dat", "w"); fstream_init(&strm, F); binstore_init(&store, &strm); test_cleanup(); @@ -36,7 +36,7 @@ static void test_group_readwrite(CuTest * tc) binstore_done(&store); fstream_done(&strm); - F = fopen("test.dat", "rb"); + F = fopen("test.dat", "r"); fstream_init(&strm, F); binstore_init(&store, &strm); f->groups = 0; diff --git a/src/kernel/save.c b/src/kernel/save.c index a4227b9ac..25ae2d858 100644 --- a/src/kernel/save.c +++ b/src/kernel/save.c @@ -256,7 +256,7 @@ int readorders(const char *filename) int nfactions = 0; struct faction *f = NULL; - F = fopen(filename, "rb"); + F = fopen(filename, "r"); if (!F) { perror(filename); return -1; @@ -1407,7 +1407,7 @@ int readgame(const char *filename, bool backup) create_backup(path); } - F = fopen(path, "rb"); + F = fopen(path, "r"); if (!F) { perror(path); return -1; @@ -1753,7 +1753,7 @@ int writegame(const char *filename) /* make sure we don't overwrite an existing file (hard links) */ unlink(path); #endif - F = fopen(path, "wb"); + F = fopen(path, "w"); if (!F) { perror(path); return -1; diff --git a/src/kernel/save.test.c b/src/kernel/save.test.c index 9e9ba9107..e7f85ad7e 100644 --- a/src/kernel/save.test.c +++ b/src/kernel/save.test.c @@ -50,7 +50,7 @@ static void test_readwrite_unit(CuTest * tc) free_gamedata(); f = test_create_faction(0); renumber_faction(f, fno); - data = gamedata_open(path, "rb"); + data = gamedata_open(path, "r"); u = read_unit(data); gamedata_close(data); diff --git a/storage b/storage index 89f3c1b01..18cc3bb8f 160000 --- a/storage +++ b/storage @@ -1 +1 @@ -Subproject commit 89f3c1b01e41f2675fcbfd51fd8494894dc22d44 +Subproject commit 18cc3bb8f8906237915eb31c9899f95340318087 From b3db14465e014b5124993cf2c53d2647b297212b Mon Sep 17 00:00:00 2001 From: Enno Rehling Date: Mon, 1 Feb 2016 13:59:35 +0100 Subject: [PATCH 14/15] fix logic error in json_buildings --- src/kernel/jsonconf.c | 6 ++++-- 1 file changed, 4 insertions(+), 2 deletions(-) diff --git a/src/kernel/jsonconf.c b/src/kernel/jsonconf.c index fc2d9dd20..099b7a228 100644 --- a/src/kernel/jsonconf.c +++ b/src/kernel/jsonconf.c @@ -114,7 +114,7 @@ static void json_maintenance_i(cJSON *json, maintenance *mt) { } break; default: - log_error("maintenance contains unknown attribute %s", child->string); + log_error("maintenance contains unknown attribute %s of type %d", child->string, child->type); } } } @@ -139,7 +139,9 @@ static void json_maintenance(cJSON *json, maintenance **mtp) { } } } - json_maintenance_i(json, mt); + else { + json_maintenance_i(json, mt); + } } static void json_construction(cJSON *json, construction **consp) { From 23e57c6bff654c3362458279cd48c7b8f32b012f Mon Sep 17 00:00:00 2001 From: Enno Rehling Date: Mon, 1 Feb 2016 14:06:56 +0100 Subject: [PATCH 15/15] fix missing translation error messages during tests. --- src/reports.test.c | 7 ++++--- src/tests.c | 2 +- 2 files changed, 5 insertions(+), 4 deletions(-) diff --git a/src/reports.test.c b/src/reports.test.c index 103dc28ff..dd2c544eb 100644 --- a/src/reports.test.c +++ b/src/reports.test.c @@ -262,6 +262,7 @@ static void test_write_unit(CuTest *tc) { locale_setstring(lang, "nr_skills", "Talente"); locale_setstring(lang, "skill::sailing", "Segeln"); locale_setstring(lang, "skill::alchemy", "Alchemie"); + locale_setstring(lang, "status_aggressive", "aggressiv"); init_skills(lang); u = test_create_unit(test_create_faction(rc), test_create_region(0, 0, 0)); u->faction->locale = lang; @@ -271,15 +272,15 @@ static void test_write_unit(CuTest *tc) { unit_setid(u, 1); bufunit(u->faction, u, 0, 0, buffer, sizeof(buffer)); - CuAssertStrEquals(tc, "Hodor (1), 1 human, status_aggressive.", buffer); + CuAssertStrEquals(tc, "Hodor (1), 1 human, aggressiv.", buffer); set_level(u, SK_SAILING, 1); bufunit(u->faction, u, 0, 0, buffer, sizeof(buffer)); - CuAssertStrEquals(tc, "Hodor (1), 1 human, status_aggressive, Talente: Segeln 1.", buffer); + CuAssertStrEquals(tc, "Hodor (1), 1 human, aggressiv, Talente: Segeln 1.", buffer); set_level(u, SK_ALCHEMY, 1); bufunit(u->faction, u, 0, 0, buffer, sizeof(buffer)); - CuAssertStrEquals(tc, "Hodor (1), 1 human, status_aggressive, Talente: Segeln 1, Alchemie 2.", buffer); + CuAssertStrEquals(tc, "Hodor (1), 1 human, aggressiv, Talente: Segeln 1, Alchemie 2.", buffer); f = test_create_faction(0); f->locale = get_or_create_locale("de"); diff --git a/src/tests.c b/src/tests.c index 0511919c3..b8508b4cc 100644 --- a/src/tests.c +++ b/src/tests.c @@ -265,6 +265,7 @@ void test_create_world(void) locale_setstring(loc, parameters[P_ANY], "ALLE"); init_parameters(loc); + locale_setstring(loc, "status_aggressive", "aggressiv"); locale_setstring(loc, keyword(K_RESERVE), "RESERVIEREN"); locale_setstring(loc, "money", "SILBER"); init_resources(); @@ -280,7 +281,6 @@ void test_create_world(void) test_create_itemtype("iron"); test_create_itemtype("stone"); - t_plain = test_create_terrain("plain", LAND_REGION | FOREST_REGION | WALK_INTO | CAVALRY_REGION | SAIL_INTO | FLY_INTO); t_plain->size = 1000; t_plain->max_road = 100;