2007-01-25 23:36:38 +01:00
|
|
|
/**
|
2007-03-22 22:50:20 +01:00
|
|
|
* vim: set ts=4 :
|
2007-01-25 23:36:38 +01:00
|
|
|
* ===============================================================
|
|
|
|
* SourceMod (C)2004-2007 AlliedModders LLC. All rights reserved.
|
|
|
|
* ===============================================================
|
|
|
|
*
|
|
|
|
* This file is not open source and may not be copied without explicit
|
|
|
|
* written permission of AlliedModders LLC. This file may not be redistributed
|
|
|
|
* in whole or significant part.
|
|
|
|
* For information, see LICENSE.txt or http://www.sourcemod.net/license.php
|
|
|
|
*
|
|
|
|
* Version: $Id$
|
|
|
|
*/
|
|
|
|
|
2007-01-16 21:36:09 +01:00
|
|
|
#include <time.h>
|
|
|
|
#include "sourcemod.h"
|
|
|
|
#include "sourcemm_api.h"
|
2007-04-05 07:25:11 +02:00
|
|
|
#include "sm_stringutil.h"
|
2007-03-10 22:26:04 +01:00
|
|
|
#include "Logger.h"
|
2007-01-25 10:19:38 +01:00
|
|
|
#include "systems/LibrarySys.h"
|
2007-01-16 21:36:09 +01:00
|
|
|
#include "sm_version.h"
|
2007-01-05 00:41:51 +01:00
|
|
|
|
2007-03-15 05:45:17 +01:00
|
|
|
Logger g_Logger;
|
2007-01-05 00:41:51 +01:00
|
|
|
|
2007-01-25 07:19:17 +01:00
|
|
|
/**
|
|
|
|
* :TODO: This should be creating the log folder if it doesn't exist
|
|
|
|
*/
|
|
|
|
|
2007-04-05 07:25:11 +02:00
|
|
|
ConfigResult Logger::OnSourceModConfigChanged(const char *key,
|
|
|
|
const char *value,
|
|
|
|
ConfigSource source,
|
|
|
|
char *error,
|
|
|
|
size_t maxlength)
|
2007-04-05 05:02:00 +02:00
|
|
|
{
|
2007-04-05 07:25:11 +02:00
|
|
|
if (strcasecmp(key, "Logging") == 0)
|
2007-04-05 05:02:00 +02:00
|
|
|
{
|
2007-04-05 12:55:40 +02:00
|
|
|
bool state;
|
2007-04-05 05:02:00 +02:00
|
|
|
|
|
|
|
if (strcasecmp(value, "on") == 0)
|
|
|
|
{
|
2007-04-05 12:55:40 +02:00
|
|
|
state = true;
|
2007-04-05 05:02:00 +02:00
|
|
|
} else if (strcasecmp(value, "off") == 0) {
|
|
|
|
state = false;
|
|
|
|
} else {
|
2007-04-05 07:25:11 +02:00
|
|
|
UTIL_Format(error, maxlength, "Invalid value: must be \"on\" or \"off\"");
|
|
|
|
return ConfigResult_Reject;
|
2007-04-05 05:02:00 +02:00
|
|
|
}
|
|
|
|
|
2007-04-05 07:25:11 +02:00
|
|
|
if (source == ConfigSource_Console)
|
2007-04-05 05:02:00 +02:00
|
|
|
{
|
2007-04-05 12:55:40 +02:00
|
|
|
state ? EnableLogging() : DisableLogging();
|
2007-04-05 05:02:00 +02:00
|
|
|
} else {
|
2007-04-05 07:25:11 +02:00
|
|
|
m_InitialState = state;
|
2007-04-05 05:02:00 +02:00
|
|
|
}
|
|
|
|
|
2007-04-05 07:25:11 +02:00
|
|
|
return ConfigResult_Accept;
|
|
|
|
} else if (strcasecmp(key, "LogMode") == 0) {
|
2007-04-05 05:02:00 +02:00
|
|
|
if (strcasecmp(value, "daily") == 0)
|
|
|
|
{
|
|
|
|
m_Mode = LoggingMode_Daily;
|
|
|
|
} else if (strcasecmp(value, "map") == 0) {
|
|
|
|
m_Mode = LoggingMode_PerMap;
|
|
|
|
} else if (strcasecmp(value, "game") == 0) {
|
|
|
|
m_Mode = LoggingMode_Game;
|
|
|
|
} else {
|
2007-04-05 07:25:11 +02:00
|
|
|
UTIL_Format(error, maxlength, "Invalid value: must be [daily|map|game]");
|
|
|
|
return ConfigResult_Reject;
|
2007-04-05 05:02:00 +02:00
|
|
|
}
|
|
|
|
|
2007-04-05 07:25:11 +02:00
|
|
|
return ConfigResult_Accept;
|
2007-04-05 05:02:00 +02:00
|
|
|
}
|
|
|
|
|
2007-04-05 07:25:11 +02:00
|
|
|
return ConfigResult_Ignore;
|
2007-04-05 05:02:00 +02:00
|
|
|
}
|
|
|
|
|
2007-03-15 05:45:17 +01:00
|
|
|
void Logger::OnSourceModStartup(bool late)
|
2007-01-11 02:11:24 +01:00
|
|
|
{
|
2007-04-05 07:25:11 +02:00
|
|
|
InitLogger(m_Mode);
|
2007-01-11 02:11:24 +01:00
|
|
|
}
|
|
|
|
|
2007-03-15 05:45:17 +01:00
|
|
|
void Logger::OnSourceModAllShutdown()
|
2007-01-11 02:11:24 +01:00
|
|
|
{
|
|
|
|
CloseLogger();
|
|
|
|
}
|
|
|
|
|
2007-04-07 05:58:20 +02:00
|
|
|
void Logger::OnSourceModLevelChange(const char *mapName)
|
|
|
|
{
|
|
|
|
MapChange(mapName);
|
|
|
|
}
|
|
|
|
|
2007-03-15 05:45:17 +01:00
|
|
|
void Logger::_NewMapFile()
|
2007-01-05 00:41:51 +01:00
|
|
|
{
|
|
|
|
if (!m_Active)
|
|
|
|
{
|
|
|
|
return;
|
|
|
|
}
|
2007-04-05 05:02:00 +02:00
|
|
|
|
|
|
|
/* Append "Log file closed" to previous log file */
|
|
|
|
_CloseFile();
|
2007-01-05 00:41:51 +01:00
|
|
|
|
|
|
|
char _filename[256];
|
|
|
|
int i = 0;
|
|
|
|
|
|
|
|
time_t t;
|
|
|
|
time(&t);
|
|
|
|
tm *curtime = localtime(&t);
|
|
|
|
|
|
|
|
while (true)
|
|
|
|
{
|
2007-01-25 07:19:17 +01:00
|
|
|
g_SourceMod.BuildPath(Path_SM, _filename, sizeof(_filename), "logs/logs_%02d%02d%03d.log", curtime->tm_mon + 1, curtime->tm_mday, i);
|
2007-01-05 00:41:51 +01:00
|
|
|
FILE *fp = fopen(_filename, "r");
|
|
|
|
if (!fp)
|
|
|
|
{
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
fclose(fp);
|
|
|
|
i++;
|
|
|
|
}
|
|
|
|
m_NrmFileName.assign(_filename);
|
|
|
|
|
|
|
|
FILE *fp = fopen(m_NrmFileName.c_str(), "w");
|
|
|
|
if (!fp)
|
|
|
|
{
|
2007-01-17 07:49:40 +01:00
|
|
|
g_SMAPI->ConPrint("[SM] Unexpected fatal logging error. SourceMod logging disabled.\n");
|
2007-01-05 00:41:51 +01:00
|
|
|
m_Active = false;
|
2007-01-07 02:30:28 +01:00
|
|
|
return;
|
2007-01-05 00:41:51 +01:00
|
|
|
} else {
|
|
|
|
char date[32];
|
|
|
|
strftime(date, sizeof(date), "%m/%d/%Y - %H:%M:%S", curtime);
|
2007-03-03 08:50:01 +01:00
|
|
|
fprintf(fp, "L %s: SourceMod log file started (file \"logs_%02d%02d%03d.log\") (Version \"%s\")\n", date, curtime->tm_mon + 1, curtime->tm_mday, i, SVN_FULL_VERSION);
|
2007-01-05 00:41:51 +01:00
|
|
|
fclose(fp);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2007-03-15 05:45:17 +01:00
|
|
|
void Logger::_CloseFile()
|
2007-01-05 00:41:51 +01:00
|
|
|
{
|
|
|
|
if (!m_Active)
|
|
|
|
{
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
FILE *fp = NULL;
|
|
|
|
if (!m_NrmFileName.empty())
|
|
|
|
{
|
|
|
|
fp = fopen(m_NrmFileName.c_str(), "r+");
|
|
|
|
if (fp)
|
|
|
|
{
|
|
|
|
fseek(fp, 0, SEEK_END);
|
|
|
|
LogMessage("Log file closed.");
|
|
|
|
fclose(fp);
|
|
|
|
}
|
|
|
|
m_NrmFileName.clear();
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!m_ErrMapStart)
|
|
|
|
{
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
fp = fopen(m_ErrFileName.c_str(), "r+");
|
|
|
|
if (fp)
|
|
|
|
{
|
|
|
|
fseek(fp, 0, SEEK_END);
|
2007-01-06 08:53:34 +01:00
|
|
|
LogError("Error log file session closed.");
|
2007-01-05 00:41:51 +01:00
|
|
|
fclose(fp);
|
|
|
|
}
|
|
|
|
m_ErrFileName.clear();
|
|
|
|
}
|
|
|
|
|
2007-04-05 07:25:11 +02:00
|
|
|
void Logger::InitLogger(LoggingMode mode)
|
2007-01-05 00:41:51 +01:00
|
|
|
{
|
2007-04-05 05:02:00 +02:00
|
|
|
m_Mode = mode;
|
2007-04-05 07:25:11 +02:00
|
|
|
m_Active = m_InitialState;
|
2007-01-05 00:41:51 +01:00
|
|
|
|
|
|
|
time_t t;
|
|
|
|
time(&t);
|
|
|
|
tm *curtime = localtime(&t);
|
|
|
|
m_CurDay = curtime->tm_mday;
|
|
|
|
|
|
|
|
char _filename[256];
|
2007-01-19 03:14:50 +01:00
|
|
|
g_SourceMod.BuildPath(Path_SM, _filename, sizeof(_filename), "logs/errors_%02d%02d%02d.log", curtime->tm_mon + 1, curtime->tm_mday, curtime->tm_year - 100);
|
2007-01-05 00:41:51 +01:00
|
|
|
m_ErrFileName.assign(_filename);
|
|
|
|
|
2007-04-05 05:02:00 +02:00
|
|
|
switch (m_Mode)
|
2007-01-05 00:41:51 +01:00
|
|
|
{
|
|
|
|
case LoggingMode_PerMap:
|
|
|
|
{
|
2007-04-05 07:25:11 +02:00
|
|
|
if (!m_Active)
|
2007-01-05 00:41:51 +01:00
|
|
|
{
|
|
|
|
m_DelayedStart = true;
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
case LoggingMode_Daily:
|
|
|
|
{
|
2007-01-25 07:19:17 +01:00
|
|
|
g_SourceMod.BuildPath(Path_SM, _filename, sizeof(_filename), "logs/logs_%02d%02d.log", curtime->tm_mon + 1, curtime->tm_mday);
|
2007-01-05 00:41:51 +01:00
|
|
|
m_NrmFileName.assign(_filename);
|
|
|
|
m_DailyPrintHdr = true;
|
|
|
|
break;
|
|
|
|
}
|
2007-01-25 10:19:38 +01:00
|
|
|
default:
|
|
|
|
{
|
|
|
|
/* do nothing... */
|
|
|
|
break;
|
|
|
|
}
|
2007-01-05 00:41:51 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2007-03-15 05:45:17 +01:00
|
|
|
void Logger::CloseLogger()
|
2007-01-05 00:41:51 +01:00
|
|
|
{
|
|
|
|
_CloseFile();
|
|
|
|
}
|
|
|
|
|
2007-03-15 05:45:17 +01:00
|
|
|
void Logger::LogMessage(const char *vafmt, ...)
|
2007-01-05 00:41:51 +01:00
|
|
|
{
|
|
|
|
if (!m_Active)
|
|
|
|
{
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2007-04-05 05:02:00 +02:00
|
|
|
if (m_Mode == LoggingMode_Game)
|
2007-01-05 00:41:51 +01:00
|
|
|
{
|
2007-01-07 02:30:28 +01:00
|
|
|
va_list ap;
|
|
|
|
va_start(ap, vafmt);
|
2007-04-05 05:02:00 +02:00
|
|
|
_PrintToGameLog(vafmt, ap);
|
2007-01-07 02:30:28 +01:00
|
|
|
va_end(ap);
|
2007-01-05 00:41:51 +01:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (m_DelayedStart)
|
|
|
|
{
|
|
|
|
m_DelayedStart = false;
|
|
|
|
_NewMapFile();
|
|
|
|
}
|
|
|
|
|
2007-01-07 02:30:28 +01:00
|
|
|
char msg[3072];
|
2007-01-05 00:41:51 +01:00
|
|
|
va_list ap;
|
|
|
|
va_start(ap, vafmt);
|
|
|
|
vsnprintf(msg, sizeof(msg), vafmt, ap);
|
|
|
|
va_end(ap);
|
|
|
|
|
|
|
|
char date[32];
|
|
|
|
time_t t;
|
|
|
|
time(&t);
|
|
|
|
tm *curtime = localtime(&t);
|
|
|
|
strftime(date, sizeof(date), "%m/%d/%Y - %H:%M:%S", curtime);
|
|
|
|
|
|
|
|
FILE *fp = NULL;
|
2007-04-05 05:02:00 +02:00
|
|
|
if (m_Mode == LoggingMode_PerMap)
|
2007-01-05 00:41:51 +01:00
|
|
|
{
|
|
|
|
fp = fopen(m_NrmFileName.c_str(), "a+");
|
|
|
|
if (!fp)
|
|
|
|
{
|
|
|
|
_NewMapFile();
|
|
|
|
fp = fopen(m_NrmFileName.c_str(), "a+");
|
|
|
|
if (!fp)
|
|
|
|
{
|
|
|
|
goto print_error;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
if (m_CurDay != curtime->tm_mday)
|
|
|
|
{
|
|
|
|
char _filename[256];
|
2007-01-25 07:19:17 +01:00
|
|
|
g_SourceMod.BuildPath(Path_SM, _filename, sizeof(_filename), "logs/logs_%02d%02d.log", curtime->tm_mon + 1, curtime->tm_mday);
|
2007-01-05 00:41:51 +01:00
|
|
|
m_NrmFileName.assign(_filename);
|
|
|
|
m_CurDay = curtime->tm_mday;
|
|
|
|
m_DailyPrintHdr = true;
|
|
|
|
}
|
|
|
|
fp = fopen(m_NrmFileName.c_str(), "a+");
|
|
|
|
}
|
|
|
|
|
|
|
|
if (fp)
|
|
|
|
{
|
|
|
|
if (m_DailyPrintHdr)
|
|
|
|
{
|
|
|
|
m_DailyPrintHdr = false;
|
2007-03-03 08:50:01 +01:00
|
|
|
fprintf(fp, "L %s: SourceMod log file session started (file \"logs_%02d%02d.log\") (Version \"%s\")\n", date, curtime->tm_mon + 1, curtime->tm_mday, SVN_FULL_VERSION);
|
|
|
|
}
|
2007-01-05 00:41:51 +01:00
|
|
|
fprintf(fp, "L %s: %s\n", date, msg);
|
|
|
|
fclose(fp);
|
|
|
|
} else {
|
|
|
|
goto print_error;
|
|
|
|
}
|
|
|
|
|
|
|
|
g_SMAPI->ConPrintf("L %s: %s\n", date, msg);
|
2007-01-05 14:23:25 +01:00
|
|
|
return;
|
2007-01-05 00:41:51 +01:00
|
|
|
print_error:
|
2007-01-17 07:49:40 +01:00
|
|
|
g_SMAPI->ConPrint("[SM] Unexpected fatal logging error. SourceMod logging disabled.\n");
|
2007-01-05 00:41:51 +01:00
|
|
|
m_Active = false;
|
|
|
|
}
|
|
|
|
|
2007-03-15 05:45:17 +01:00
|
|
|
void Logger::LogError(const char *vafmt, ...)
|
2007-01-05 00:41:51 +01:00
|
|
|
{
|
|
|
|
if (!m_Active)
|
|
|
|
{
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
time_t t;
|
|
|
|
time(&t);
|
|
|
|
tm *curtime = localtime(&t);
|
|
|
|
|
|
|
|
char date[32];
|
|
|
|
strftime(date, sizeof(date), "%m/%d/%Y - %H:%M:%S", curtime);
|
|
|
|
|
|
|
|
if (curtime->tm_mday != m_CurDay)
|
|
|
|
{
|
|
|
|
char _filename[256];
|
2007-01-19 03:14:50 +01:00
|
|
|
g_SourceMod.BuildPath(Path_SM, _filename, sizeof(_filename), "logs/errors_%02d%02d%02d.log", curtime->tm_mon + 1, curtime->tm_mday, curtime->tm_year - 100);
|
2007-01-05 00:41:51 +01:00
|
|
|
m_ErrFileName.assign(_filename);
|
|
|
|
m_CurDay = curtime->tm_mday;
|
|
|
|
m_ErrMapStart = false;
|
|
|
|
}
|
|
|
|
|
2007-01-07 02:30:28 +01:00
|
|
|
char msg[3072];
|
2007-01-05 00:41:51 +01:00
|
|
|
va_list ap;
|
|
|
|
va_start(ap, vafmt);
|
|
|
|
vsnprintf(msg, sizeof(msg), vafmt, ap);
|
|
|
|
va_end(ap);
|
|
|
|
|
|
|
|
FILE *fp = fopen(m_ErrFileName.c_str(), "a+");
|
|
|
|
if (fp)
|
|
|
|
{
|
|
|
|
if (!m_ErrMapStart)
|
|
|
|
{
|
|
|
|
fprintf(fp, "L %s: SourceMod error session started\n", date);
|
|
|
|
fprintf(fp, "L %s: Info (map \"%s\") (log file \"errors_%02d%02d%02d.log\")\n", date, m_CurMapName.c_str(), curtime->tm_mon + 1, curtime->tm_mday, curtime->tm_year - 100);
|
|
|
|
m_ErrMapStart = true;
|
|
|
|
}
|
|
|
|
fprintf(fp, "L %s: %s\n", date, msg);
|
|
|
|
fclose(fp);
|
|
|
|
} else {
|
2007-01-17 07:49:40 +01:00
|
|
|
g_SMAPI->ConPrint("[SM] Unexpected fatal logging error. SourceMod logging disabled.\n");
|
2007-01-05 00:41:51 +01:00
|
|
|
m_Active = false;
|
2007-01-07 02:30:28 +01:00
|
|
|
return;
|
2007-01-05 00:41:51 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
g_SMAPI->ConPrintf("L %s: %s\n", date, msg);
|
|
|
|
}
|
|
|
|
|
2007-03-15 05:45:17 +01:00
|
|
|
void Logger::MapChange(const char *mapname)
|
2007-01-05 00:41:51 +01:00
|
|
|
{
|
|
|
|
m_CurMapName.assign(mapname);
|
|
|
|
|
2007-04-05 05:02:00 +02:00
|
|
|
switch (m_Mode)
|
2007-01-05 00:41:51 +01:00
|
|
|
{
|
|
|
|
case LoggingMode_Daily:
|
|
|
|
{
|
|
|
|
LogMessage("-------- Mapchange to %s --------", mapname);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
case LoggingMode_PerMap:
|
|
|
|
{
|
|
|
|
_NewMapFile();
|
|
|
|
break;
|
|
|
|
}
|
2007-01-25 10:19:38 +01:00
|
|
|
default:
|
|
|
|
{
|
|
|
|
/* Do nothing... */
|
|
|
|
break;
|
|
|
|
}
|
2007-01-05 00:41:51 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
if (m_ErrMapStart)
|
|
|
|
{
|
2007-01-06 08:53:34 +01:00
|
|
|
LogError("Error log file session closed.");
|
2007-01-05 00:41:51 +01:00
|
|
|
}
|
|
|
|
m_ErrMapStart = false;
|
|
|
|
}
|
|
|
|
|
2007-04-05 05:02:00 +02:00
|
|
|
void Logger::_PrintToGameLog(const char *fmt, va_list ap)
|
2007-01-05 00:41:51 +01:00
|
|
|
{
|
2007-01-07 02:30:28 +01:00
|
|
|
char msg[3072];
|
2007-01-05 00:41:51 +01:00
|
|
|
size_t len;
|
|
|
|
|
2007-01-07 02:30:28 +01:00
|
|
|
len = vsnprintf(msg, sizeof(msg)-2, fmt, ap);
|
|
|
|
len = (len >= sizeof(msg)) ? (sizeof(msg) - 2) : len;
|
|
|
|
|
2007-01-05 00:41:51 +01:00
|
|
|
msg[len++] = '\n';
|
|
|
|
msg[len] = '\0';
|
|
|
|
|
|
|
|
engine->LogPrint(msg);
|
2007-01-16 21:36:09 +01:00
|
|
|
}
|
|
|
|
|
2007-03-15 05:45:17 +01:00
|
|
|
const char *Logger::GetLogFileName(LogType type) const
|
2007-01-16 21:36:09 +01:00
|
|
|
{
|
|
|
|
switch (type)
|
|
|
|
{
|
|
|
|
case LogType_Normal:
|
|
|
|
{
|
|
|
|
return m_NrmFileName.c_str();
|
|
|
|
}
|
|
|
|
case LogType_Error:
|
|
|
|
{
|
|
|
|
return m_ErrFileName.c_str();
|
|
|
|
}
|
|
|
|
default:
|
|
|
|
{
|
|
|
|
return "";
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2007-03-15 05:45:17 +01:00
|
|
|
LoggingMode Logger::GetLoggingMode() const
|
2007-01-16 21:36:09 +01:00
|
|
|
{
|
2007-04-05 05:02:00 +02:00
|
|
|
return m_Mode;
|
2007-01-16 21:36:09 +01:00
|
|
|
}
|
|
|
|
|
2007-03-15 05:45:17 +01:00
|
|
|
void Logger::EnableLogging()
|
2007-01-16 21:36:09 +01:00
|
|
|
{
|
|
|
|
if (m_Active)
|
|
|
|
{
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
m_Active = true;
|
2007-04-05 12:55:40 +02:00
|
|
|
LogMessage("[SM] Logging enabled manually by user.");
|
2007-01-16 21:36:09 +01:00
|
|
|
}
|
|
|
|
|
2007-03-15 05:45:17 +01:00
|
|
|
void Logger::DisableLogging()
|
2007-01-16 21:36:09 +01:00
|
|
|
{
|
|
|
|
if (!m_Active)
|
|
|
|
{
|
|
|
|
return;
|
|
|
|
}
|
2007-04-05 12:55:40 +02:00
|
|
|
LogMessage("[SM] Logging disabled manually by user.");
|
2007-01-16 21:36:09 +01:00
|
|
|
m_Active = false;
|
|
|
|
}
|
2007-04-05 07:25:11 +02:00
|
|
|
|
|
|
|
void Logger::LogFatal(const char *msg, ...)
|
|
|
|
{
|
2007-04-05 07:45:35 +02:00
|
|
|
/* :TODO: make this print all pretty-like
|
|
|
|
* In fact, the pretty log printing function should be abstracted.
|
|
|
|
* It's already implemented twice which is bad.
|
|
|
|
*/
|
|
|
|
|
2007-04-05 07:25:11 +02:00
|
|
|
char path[PLATFORM_MAX_PATH];
|
|
|
|
g_SourceMod.BuildPath(Path_Game, path, sizeof(path), "sourcemod_fatal.log");
|
|
|
|
FILE *fp = fopen(path, "at");
|
|
|
|
if (!fp)
|
|
|
|
{
|
|
|
|
/* We're just doomed, aren't we... */
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
va_list ap;
|
|
|
|
va_start(ap, msg);
|
|
|
|
vfprintf(fp, msg, ap);
|
|
|
|
va_end(ap);
|
|
|
|
|
|
|
|
fputs("\n", fp);
|
|
|
|
|
|
|
|
fclose(fp);
|
|
|
|
}
|