mirror of
https://github.com/matrix-construct/construct
synced 2025-01-19 11:01:54 +01:00
507 lines
13 KiB
C++
507 lines
13 KiB
C++
/*
|
|
* charybdis: an advanced Internet Relay Chat Daemon(ircd).
|
|
*
|
|
* Copyright (C) 2003 Lee H <lee@leeh.co.uk>
|
|
* Copyright (C) 2003-2005 ircd-ratbox development team
|
|
* Copyright (C) 2008 William Pitcock <nenolod@sacredspiral.co.uk>
|
|
* Copyright (C) 2016 Charybdis Development Team
|
|
* Copyright (C) 2016 Jason Volk <jason@zemos.net>
|
|
*
|
|
* Redistribution and use in source and binary forms, with or without
|
|
* modification, are permitted provided that the following conditions are
|
|
* met:
|
|
*
|
|
* 1.Redistributions of source code must retain the above copyright notice,
|
|
* this list of conditions and the following disclaimer.
|
|
*
|
|
* 2.Redistributions in binary form must reproduce the above copyright
|
|
* notice, this list of conditions and the following disclaimer in the
|
|
* documentation and/or other materials provided with the distribution.
|
|
*
|
|
* 3.The name of the author may not be used to endorse or promote products
|
|
* derived from this software without specific prior written permission.
|
|
*
|
|
* THIS SOFTWARE IS PROVIDED BY THE AUTHOR ``AS IS'' AND ANY EXPRESS OR
|
|
* IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED
|
|
* WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE
|
|
* DISCLAIMED. IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR ANY DIRECT,
|
|
* INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES
|
|
* (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR
|
|
* SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
|
|
* HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT,
|
|
* STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING
|
|
* IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE
|
|
* POSSIBILITY OF SUCH DAMAGE.
|
|
*/
|
|
|
|
namespace ircd::log
|
|
{
|
|
// Option toggles
|
|
std::array<bool, num_of<facility>()> file_flush;
|
|
std::array<bool, num_of<facility>()> console_flush;
|
|
std::array<const char *, num_of<facility>()> console_ansi;
|
|
|
|
// Runtime master switches
|
|
std::array<bool, num_of<facility>()> file_out;
|
|
std::array<bool, num_of<facility>()> console_out;
|
|
std::array<bool, num_of<facility>()> console_err;
|
|
|
|
// Suppression state (for struct console_quiet)
|
|
std::array<bool, num_of<facility>()> quieted_out;
|
|
std::array<bool, num_of<facility>()> quieted_err;
|
|
|
|
// Logfile name and device
|
|
std::array<const char *, num_of<facility>()> fname;
|
|
std::array<std::ofstream, num_of<facility>()> file;
|
|
|
|
std::ostream &out_console
|
|
{
|
|
std::cout
|
|
};
|
|
|
|
std::ostream &err_console
|
|
{
|
|
std::cerr
|
|
};
|
|
|
|
/*
|
|
ConfEntry conf_log_table[] =
|
|
{
|
|
{ "file_critical", CF_QSTRING, NULL, PATH_MAX, &fname[CRITICAL] },
|
|
{ "file_error", CF_QSTRING, NULL, PATH_MAX, &fname[ERROR] },
|
|
{ "file_warning", CF_QSTRING, NULL, PATH_MAX, &fname[WARNING] },
|
|
{ "file_notice", CF_QSTRING, NULL, PATH_MAX, &fname[NOTICE] },
|
|
{ "file_info", CF_QSTRING, NULL, PATH_MAX, &fname[INFO] },
|
|
{ "file_debug", CF_QSTRING, NULL, PATH_MAX, &fname[DEBUG] },
|
|
};
|
|
*/
|
|
|
|
static void open(const facility &fac);
|
|
static void slog(const facility &fac, const std::function<void (mutable_buffer &)> &closure) noexcept;
|
|
static void vlog_threadsafe(const facility &fac, const std::string &name, const char *const &fmt, const va_rtti &ap);
|
|
}
|
|
|
|
void
|
|
ircd::log::init()
|
|
{
|
|
//TODO: XXX: config + cmd control + other fancy stuff
|
|
|
|
//add_top_conf("log", NULL, NULL, conf_log_table);
|
|
|
|
console_err[CRITICAL] = true;
|
|
console_err[ERROR] = true;
|
|
console_err[WARNING] = true;
|
|
|
|
console_out[NOTICE] = true;
|
|
console_out[INFO] = true;
|
|
console_out[DEBUG] = ircd::debugmode;
|
|
|
|
file_out[CRITICAL] = true;
|
|
file_out[ERROR] = true;
|
|
file_out[WARNING] = true;
|
|
file_out[NOTICE] = true;
|
|
file_out[INFO] = true;
|
|
file_out[DEBUG] = ircd::debugmode;
|
|
|
|
file_flush[CRITICAL] = true;
|
|
file_flush[ERROR] = true;
|
|
file_flush[WARNING] = true;
|
|
file_flush[NOTICE] = false;
|
|
file_flush[INFO] = false;
|
|
file_flush[DEBUG] = false;
|
|
|
|
console_flush[CRITICAL] = true;
|
|
console_flush[ERROR] = true;
|
|
console_flush[WARNING] = true;
|
|
console_flush[NOTICE] = false;
|
|
console_flush[INFO] = false;
|
|
console_flush[DEBUG] = true;
|
|
|
|
console_ansi[CRITICAL] = "\033[1;5;37;45m";
|
|
console_ansi[ERROR] = "\033[1;37;41m";
|
|
console_ansi[WARNING] = "\033[0;30;43m";
|
|
console_ansi[NOTICE] = "\033[1;37;46m";
|
|
console_ansi[INFO] = "\033[1;37;42m";
|
|
console_ansi[DEBUG] = "\033[1;30;47m";
|
|
}
|
|
|
|
void
|
|
ircd::log::fini()
|
|
{
|
|
//remove_top_conf("log");
|
|
}
|
|
|
|
void
|
|
ircd::log::open()
|
|
{
|
|
for_each<facility>([](const facility &fac)
|
|
{
|
|
if(!fname[fac])
|
|
return;
|
|
|
|
if(!file_out[fac])
|
|
return;
|
|
|
|
if(file[fac].is_open())
|
|
file[fac].close();
|
|
|
|
file[fac].clear();
|
|
file[fac].exceptions(std::ios::badbit | std::ios::failbit);
|
|
open(fac);
|
|
});
|
|
}
|
|
|
|
void
|
|
ircd::log::close()
|
|
{
|
|
for_each<facility>([](const facility &fac)
|
|
{
|
|
if(file[fac].is_open())
|
|
file[fac].close();
|
|
});
|
|
}
|
|
|
|
void
|
|
ircd::log::flush()
|
|
{
|
|
for_each<facility>([](const facility &fac)
|
|
{
|
|
file[fac].flush();
|
|
});
|
|
}
|
|
|
|
void
|
|
ircd::log::open(const facility &fac)
|
|
try
|
|
{
|
|
const auto &mode(std::ios::app);
|
|
file[fac].open(fname[fac], mode);
|
|
}
|
|
catch(const std::exception &e)
|
|
{
|
|
char buf[BUFSIZE];
|
|
snprintf(buf, sizeof(buf), "!!! Opening log file [%s] failed: %s",
|
|
fname[fac],
|
|
e.what());
|
|
|
|
std::cerr << buf << std::endl;
|
|
throw;
|
|
}
|
|
|
|
ircd::log::console_quiet::console_quiet(const bool &showmsg)
|
|
{
|
|
if(showmsg)
|
|
notice("Log messages are now quieted at the console");
|
|
|
|
std::copy(begin(console_out), end(console_out), begin(quieted_out));
|
|
std::copy(begin(console_err), end(console_err), begin(quieted_err));
|
|
std::fill(begin(console_out), end(console_out), false);
|
|
std::fill(begin(console_err), end(console_err), false);
|
|
|
|
// Make a special amend to never suppress CRITICAL messages because
|
|
// these are usually for a crash or major b0rk where the console
|
|
// user probably won't be continuing normally anyway...
|
|
if(quieted_out[CRITICAL]) console_out[CRITICAL] = true;
|
|
if(quieted_err[CRITICAL]) console_err[CRITICAL] = true;
|
|
}
|
|
|
|
ircd::log::console_quiet::~console_quiet()
|
|
{
|
|
std::copy(begin(quieted_out), end(quieted_out), begin(console_out));
|
|
std::copy(begin(quieted_err), end(quieted_err), begin(console_err));
|
|
|
|
std::cout << std::flush;
|
|
std::cout.clear();
|
|
std::cerr.clear();
|
|
}
|
|
|
|
ircd::log::log::log(const std::string &name)
|
|
:name{name}
|
|
{
|
|
}
|
|
|
|
ircd::log::log::log(const std::string &name,
|
|
const char &snote)
|
|
:log{name}
|
|
{
|
|
}
|
|
|
|
void
|
|
ircd::log::mark(const char *const &msg)
|
|
{
|
|
for_each<facility>([&msg]
|
|
(const auto &fac)
|
|
{
|
|
mark(fac, msg);
|
|
});
|
|
}
|
|
|
|
void
|
|
ircd::log::mark(const facility &fac,
|
|
const char *const &msg)
|
|
{
|
|
static const auto name{"*"s};
|
|
vlog(fac, name, "%s", msg);
|
|
}
|
|
|
|
namespace ircd::log
|
|
{
|
|
static void check(std::ostream &) noexcept;
|
|
static mutable_buffer compose(mutable_buffer &out, const string_view &buf, const string_view &name);
|
|
}
|
|
|
|
/// ircd::log is not thread-safe. This internal function is called when the
|
|
/// normal vlog() detects it's not on the main IRCd thread. It then generates
|
|
/// the formatted log message on this thread, and posts the message to the
|
|
/// main IRCd event loop which is running on the main thread.
|
|
void
|
|
ircd::log::vlog_threadsafe(const facility &fac,
|
|
const std::string &name,
|
|
const char *const &fmt,
|
|
const va_rtti &ap)
|
|
{
|
|
// Generate the formatted message on this thread first
|
|
std::string str
|
|
{
|
|
fmt::vsnstringf(1024, fmt, ap)
|
|
};
|
|
|
|
// The reference to name has to be safely maintained
|
|
ircd::post([fac, str(std::move(str)), name(std::move(name))]
|
|
{
|
|
slog(fac, [&str, &name](mutable_buffer &out)
|
|
{
|
|
compose(out, str, name);
|
|
});
|
|
});
|
|
}
|
|
|
|
void
|
|
ircd::log::vlog(const facility &fac,
|
|
const char *const &fmt,
|
|
const va_rtti &ap)
|
|
{
|
|
static const auto name{"ircd"s};
|
|
vlog(fac, name, fmt, ap);
|
|
}
|
|
|
|
void
|
|
ircd::log::vlog(const facility &fac,
|
|
const std::string &name,
|
|
const char *const &fmt,
|
|
const va_rtti &ap)
|
|
{
|
|
if(!is_main_thread())
|
|
{
|
|
vlog_threadsafe(fac, name, fmt, ap);
|
|
return;
|
|
}
|
|
|
|
char buf[1024];
|
|
const auto len
|
|
{
|
|
fmt::vsprintf(buf, fmt, ap)
|
|
};
|
|
|
|
const string_view msg{buf, size_t(len)};
|
|
slog(fac, [&msg, &name](mutable_buffer &out)
|
|
{
|
|
compose(out, msg, name);
|
|
});
|
|
}
|
|
|
|
void
|
|
ircd::log::slog(const facility &fac,
|
|
const std::function<void (mutable_buffer &)> &closure)
|
|
noexcept
|
|
{
|
|
if(!file[fac].is_open() && !console_out[fac] && !console_err[fac])
|
|
return;
|
|
|
|
// Have to be on the main thread to call slog(). If slog() yields for some
|
|
// reason it's a problem too. During the composition of this log message,
|
|
// if another log message is created from calls for normal reasons or from
|
|
// errors, that's not good either. We can only have one log slog() at a
|
|
// time for now...
|
|
assert_main_thread();
|
|
const ctx::critical_assertion ca;
|
|
static bool entered;
|
|
assert(!entered);
|
|
entered = true;
|
|
const unwind leaving([]
|
|
{
|
|
entered = false;
|
|
});
|
|
|
|
char buf[1024];
|
|
static const string_view terminator{"\r\n"};
|
|
const size_t max(sizeof(buf) - size(terminator));
|
|
|
|
std::stringstream s;
|
|
s.rdbuf()->pubsetbuf(buf, max);
|
|
|
|
//TODO: XXX: Add option toggle for smalldate()
|
|
char date[64];
|
|
s << microtime(date)
|
|
<< ' '
|
|
<< (console_ansi[fac]? console_ansi[fac] : "")
|
|
<< std::setw(8)
|
|
<< std::right
|
|
<< reflect(fac)
|
|
<< (console_ansi[fac]? "\033[0m " : " ");
|
|
|
|
// We setup a buffer starting directly after the prefix and pass it
|
|
// to the user's closure. `mb` is used as a stream buffer here: as it
|
|
// is consumed its data pointer is advanced and thus its size decreases.
|
|
const size_t consumed{std::min(size_t(s.tellp()), max)};
|
|
mutable_buffer mb{buf + consumed, max - consumed};
|
|
assert(!full(mb));
|
|
closure(mb);
|
|
|
|
// In all of the above, some extra space for the newline terminator
|
|
// was left at the end; we shift the buffer window past where `mb`
|
|
// stopped and append it here.
|
|
mb = mutable_buffer{data(mb), size(terminator)};
|
|
assert(!full(mb));
|
|
consume(mb, copy(mb, terminator));
|
|
|
|
// The final output view starts at the very beginning and its size is
|
|
// determined by where the `mb` data pointer currently points.
|
|
const size_t outsz(std::distance(buf, data(mb)));
|
|
assert(outsz <= sizeof(buf));
|
|
const mutable_buffer out
|
|
{
|
|
buf, outsz
|
|
};
|
|
|
|
const auto write{[&out](std::ostream &s)
|
|
{
|
|
check(s);
|
|
s.write(data(out), size(out));
|
|
}};
|
|
|
|
// copy to std::cerr
|
|
if(console_err[fac])
|
|
{
|
|
err_console.clear();
|
|
write(err_console);
|
|
}
|
|
|
|
// copy to std::cout
|
|
if(console_out[fac])
|
|
{
|
|
out_console.clear();
|
|
write(out_console);
|
|
if(console_flush[fac])
|
|
std::flush(out_console);
|
|
}
|
|
|
|
// copy to file
|
|
if(file[fac].is_open())
|
|
{
|
|
file[fac].clear();
|
|
write(file[fac]);
|
|
if(file_flush[fac])
|
|
std::flush(file[fac]);
|
|
}
|
|
}
|
|
|
|
ircd::mutable_buffer
|
|
ircd::log::compose(mutable_buffer &out,
|
|
const string_view &buf,
|
|
const string_view &name)
|
|
{
|
|
std::stringstream s;
|
|
s.rdbuf()->pubsetbuf(data(out), size(out));
|
|
s << std::setw(9)
|
|
<< std::right
|
|
<< name
|
|
<< ' '
|
|
<< std::setw(8)
|
|
<< trunc(ctx::name(), 8)
|
|
<< ' '
|
|
<< std::setw(6)
|
|
<< std::right
|
|
<< ctx::id()
|
|
<< " :"
|
|
<< buf;
|
|
|
|
const size_t tell(s.tellp());
|
|
const size_t retsz{std::min(tell, size(out))};
|
|
const mutable_buffer ret
|
|
{
|
|
data(out), retsz
|
|
};
|
|
|
|
assert(retsz <= size(out));
|
|
consume(out, retsz);
|
|
return ret;
|
|
}
|
|
|
|
void
|
|
ircd::log::check(std::ostream &s)
|
|
noexcept try
|
|
{
|
|
if(likely(s.good()))
|
|
return;
|
|
|
|
char buf[128];
|
|
snprintf(buf, sizeof(buf), "fatal: log stream good[%d] bad[%d] fail[%d] eof[%d]",
|
|
s.good(),
|
|
s.bad(),
|
|
s.fail(),
|
|
s.eof());
|
|
|
|
fprintf(stderr, "log stream(%p) fatal: %s\n", (const void *)&s, buf);
|
|
fprintf(stdout, "log stream(%p) fatal: %s\n", (const void *)&s, buf);
|
|
fflush(stderr);
|
|
fflush(stdout);
|
|
s.exceptions(s.eofbit | s.failbit | s.badbit);
|
|
throw std::runtime_error(buf);
|
|
}
|
|
catch(const std::exception &e)
|
|
{
|
|
fprintf(stderr, "%s\n", e.what());
|
|
fprintf(stdout, "%s\n", e.what());
|
|
fflush(stderr);
|
|
fflush(stdout);
|
|
ircd::terminate();
|
|
}
|
|
|
|
const char *
|
|
ircd::log::reflect(const facility &f)
|
|
{
|
|
switch(f)
|
|
{
|
|
case facility::DEBUG: return "DEBUG";
|
|
case facility::INFO: return "INFO";
|
|
case facility::NOTICE: return "NOTICE";
|
|
case facility::WARNING: return "WARNING";
|
|
case facility::ERROR: return "ERROR";
|
|
case facility::CRITICAL: return "CRITICAL";
|
|
case facility::_NUM_: break; // Allows -Wswitch to remind developer to add reflection here
|
|
};
|
|
|
|
return "??????";
|
|
}
|
|
|
|
const char *
|
|
ircd::smalldate(const time_t <ime)
|
|
{
|
|
static const size_t MAX_DATE_STRING
|
|
{
|
|
32 // maximum string length for a date string (ircd_defs.h)
|
|
};
|
|
|
|
struct tm lt;
|
|
localtime_r(<ime, <);
|
|
static char buf[MAX_DATE_STRING];
|
|
snprintf(buf, sizeof(buf), "%d/%d/%d %02d.%02d",
|
|
lt.tm_year + 1900,
|
|
lt.tm_mon + 1,
|
|
lt.tm_mday,
|
|
lt.tm_hour,
|
|
lt.tm_min);
|
|
|
|
return buf;
|
|
}
|