/* * Empire - A multi-player, client/server Internet based war game. * Copyright (C) 1986-2009, Dave Pare, Jeff Bailey, Thomas Ruschak, * Ken Stevens, Steve McClure * * This program is free software; you can redistribute it and/or modify * it under the terms of the GNU General Public License as published by * the Free Software Foundation; either version 2 of the License, or * (at your option) any later version. * * This program is distributed in the hope that it will be useful, * but WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the * GNU General Public License for more details. * * You should have received a copy of the GNU General Public License * along with this program; if not, write to the Free Software * Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA * * --- * * See the "LEGAL", "LICENSE", "CREDITS" and "README" files for all the * related information and legal notices. It is expected that any future * projects/authors will amend these files as needed. * * --- * * journal.c: Log a journal of events to a file * * Known contributors to this file: * Markus Armbruster, 2004-2008 * Ron Koenderink, 2008 */ /* * Journal file format: each line logs an event, and looks like this: * * TIME THREAD EVENT DATA * * Events and their data are: * * startup * shutdown * prng NAME SEED * login CNUM HOSTADDR USER * logout CNUM * command NAME * input INPUT * output THREAD ID OUTPUT * update ETU */ #include #include #include #include #include #include #include "misc.h" #include "empthread.h" #include "journal.h" #include "optlist.h" #include "player.h" #include "prototypes.h" static char journal_fname[] = "journal.log"; static FILE *journal; static void journal_entry(char *fmt, ...) ATTRIBUTE((format (printf, 1, 2))); static void journal_output_1(struct player *, int, char *, char *, int); static FILE * journal_open(void) { return fopen(journal_fname, "a+"); } static void journal_entry(char *fmt, ...) { static char buf[1024]; va_list ap; time_t now; unsigned char *p; if (journal) { time(&now); fprintf(journal, "%.24s %10.10s ", ctime(&now), empth_name(empth_self())); va_start(ap, fmt); vsnprintf(buf, sizeof(buf) - 1, fmt, ap); va_end(ap); for (p = (unsigned char *)buf; *p; p++) { if (isprint(*p)) putc(*p, journal); else fprintf(journal, "\\%03o", *p); } fputs("\n", journal); if (fmt[0] != 'o') /* FIXME disgusting hack */ fflush(journal); if (ferror(journal)) { logerror("Error writing journal (%s)", strerror(errno)); clearerr(journal); } } } int journal_startup(void) { if (!keep_journal) return 0; journal = journal_open(); if (!journal) { logerror("Can't open %s (%s)", journal_fname, strerror(errno)); return -1; } journal_entry("startup"); return 0; } void journal_shutdown(void) { journal_entry("shutdown"); if (journal) { fclose(journal); journal = NULL; } } int journal_reopen(void) { FILE *j; if (!keep_journal) return 0; j = journal_open(); if (!j) { logerror("Can't open %s (%s)", journal_fname, strerror(errno)); return -1; } if (journal) fclose(journal); journal = j; return 0; } void journal_prng(unsigned seed) { journal_entry("prng BSD %d", seed); } void journal_login(void) { journal_entry("login %d %s %s", player->cnum, player->hostaddr, player->userid); } void journal_logout(void) { journal_entry("logout %d", player->cnum); } void journal_output(struct player *pl, int id, char *output) { static char buf[1024]; static struct player *bpl; static int bid; char *s, *e; if (keep_journal < 2) return; if (buf[0] && (pl != bpl && id != bid)) { journal_output_1(bpl, bid, buf, "", -1); buf[0] = 0; } for (s = output; (e = strchr(s, '\n')); s = e + 1) { journal_output_1(pl, id, buf, s, (int)(e + 1 - s)); buf[0] = 0; } if (strlen(buf) + strlen(s) < 1024) { strcpy(buf + strlen(buf), s); bpl = pl; bid = id; } else { journal_output_1(pl, id, buf, s, -1); buf[0] = 0; } } static void journal_output_1(struct player *pl, int id, char *buf1, char *buf2, int buf2prec) { journal_entry("output %s %d %s%.*s", empth_name(empth_self()), id, buf1, buf2prec, buf2); } void journal_input(char *input) { journal_entry("input %s", input); } void journal_command(char *cmd) { char *eptr = strchr(cmd, ' '); journal_entry("command %.*s", eptr ? (int)(eptr - cmd) : -1, cmd); } void journal_update(int etu) { journal_entry("update %d", etu); }