wesnoth/src/log.hpp
Pentarctagon fc7c87b765 Show the log directory in the build info on non-windows platforms.
Also moves the last bit of potentially common code from log_windows into the base logger.
2022-10-26 19:13:57 -05:00

260 lines
8.1 KiB
C++

/*
Copyright (C) 2004 - 2022
by Guillaume Melquiond <guillaume.melquiond@gmail.com>
Copyright (C) 2003 by David White <dave@whitevine.net>
Part of the Battle for Wesnoth Project https://www.wesnoth.org/
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.
See the COPYING file for more details.
*/
/**
* @file
* Standard logging facilities (interface).
*
* To use one of the standard log channels, put something like the following at the start
* of your .cpp file:
*
* static lg::log_domain log_display("display");
* \#define ERR_DP LOG_STREAM(err, log_display)
* \#define LOG_DP LOG_STREAM(info, log_display)
*
* Then stream logging info to ERR_DP, or LOG_DP, as if it were an ostream like std::cerr.
* (In general it will actually be std::cerr at runtime when logging is enabled.)
*
* LOG_DP << "Found a window resize event: ...";
*
* Please do not use iomanip features like std::hex directly on the logger. Because of the
* design of the logger, this will result in all of the loggers (in fact std::cerr) being
* imbued with std::hex. Please use a formatter instead.
*
* \#include "formatter.hpp"
*
* LOG_DP << (formatter() << "The random seed is: '" << std::hex << seed << "'\n").str();
*
* It might be nice if somehow the logger class / macros could support using iomanip
* things directly, but right now it doesn't, and it seems that it would complicate the
* design greatly enough that it doesn't seem worth it.
*/
#pragma once
#ifndef __func__
#ifdef __FUNCTION__
#define __func__ __FUNCTION__
#endif
#endif
#include <iosfwd> // needed else all files including log.hpp need to do it.
#include <sstream> // as above. iostream (actually, iosfwd) declares stringstream as an incomplete type, but does not define it
#include <string>
#include <utility>
#include <ctime>
#include "formatter.hpp"
namespace lg {
// Prefix and extension for log files. This is used both to generate the unique
// log file name during startup and to find old files to delete.
const std::string log_file_prefix = "wesnoth-";
const std::string log_file_suffix = ".log";
// Maximum number of older log files to keep intact. Other files are deleted.
// Note that this count does not include the current log file!
const unsigned max_logs = 8;
enum severity
{
LG_ERROR=0,
LG_WARN=1,
LG_INFO=2,
LG_DEBUG=3
};
/**
* Helper class to redirect the output of the logger in a certain scope.
*
* The main usage of the redirection is for the unit tests to validate the
* output on the logger with the expected output.
*/
class redirect_output_setter
{
public:
/**
* Constructor.
*
* @param stream The stream to direct the output to.
*/
explicit redirect_output_setter(std::ostream& stream);
~redirect_output_setter();
private:
/**
* The previously set redirection.
*
* This value is stored here to be restored in this destructor.
*/
std::ostream* old_stream_;
};
class logger;
typedef std::pair<const std::string, int> logd;
class log_domain {
logd *domain_;
public:
explicit log_domain(char const *name, int severity = 1);
friend class logger;
};
bool set_log_domain_severity(const std::string& name, int severity);
bool set_log_domain_severity(const std::string& name, const logger &lg);
bool get_log_domain_severity(const std::string& name, int &severity);
std::string list_logdomains(const std::string& filter);
void set_strict_severity(int severity);
void set_strict_severity(const logger &lg);
bool broke_strict();
void set_log_to_file();
bool is_not_log_file(const std::string& filename);
void rotate_logs(const std::string& log_dir);
std::string unique_log_filename();
// A little "magic" to surround the logging operation in a mutex.
// This works by capturing the output first to a stringstream formatter, then
// locking a mutex and dumping it to the stream all in one go.
// By doing this we can avoid rare deadlocks if a function whose output is streamed
// calls logging of its own.
// We overload operator| only because it has lower precedence than operator<<
// Any other lower-precedence operator would have worked just as well.
class log_in_progress {
std::ostream& stream_;
int indent_ = 0;
bool timestamp_ = false;
std::string prefix_;
bool auto_newline_ = true;
public:
log_in_progress(std::ostream& stream);
void operator|(formatter&& message);
void set_indent(int level);
void enable_timestamp();
void set_prefix(const std::string& prefix);
void set_auto_newline(bool enabled);
};
class logger {
char const *name_;
int severity_;
public:
logger(char const *name, int severity): name_(name), severity_(severity) {}
log_in_progress operator()(const log_domain& domain,
bool show_names = true, bool do_indent = false, bool show_timestamps = true, bool break_strict = true, bool auto_newline = true) const;
bool dont_log(const log_domain& domain) const
{
return severity_ > domain.domain_->second;
}
/**
* Returns following values depending on the logger:
* error: 0
* warn: 1
* info: 2
* debug: 3
* See also the lg::severity enum.
*/
int get_severity() const
{
return severity_;
}
std::string get_name() const
{
return name_;
}
};
void timestamps(bool);
void precise_timestamps(bool);
std::string get_timestamp(const std::time_t& t, const std::string& format="%Y%m%d %H:%M:%S ");
std::string get_timespan(const std::time_t& t);
std::string sanitize_log(const std::string& logstr);
std::string& get_log_file_path();
void set_log_file_path(const std::string& path);
logger &err(), &warn(), &info(), &debug();
log_domain& general();
class scope_logger
{
int64_t ticks_;
const log_domain& domain_;
std::string str_;
public:
scope_logger(const log_domain& domain, const char* str)
: ticks_(0)
, domain_(domain)
, str_()
{
if (!debug().dont_log(domain)) do_log_entry(str);
}
scope_logger(const log_domain& domain, const std::string& str)
: ticks_(0)
, domain_(domain)
, str_()
{
if (!debug().dont_log(domain)) do_log_entry(str);
}
~scope_logger()
{
if (!str_.empty()) do_log_exit();
}
private:
void do_log_entry(const std::string& str) noexcept;
void do_log_exit() noexcept;
};
/**
* Use this to show WML errors in the ingame chat.
* After every WML event the errors are shown to the user so they can inform the campaign maintainer.
*/
std::stringstream& log_to_chat();
} // namespace lg
#define log_scope(description) lg::scope_logger scope_logging_object__(lg::general(), description);
#define log_scope2(domain,description) lg::scope_logger scope_logging_object__(domain, description);
#define LOG_STREAM(level, domain) if (lg::level().dont_log(domain)) ; else lg::level()(domain) | formatter()
// Don't prefix the logdomain to messages on this stream
#define LOG_STREAM_NAMELESS(level, domain) if (lg::level().dont_log(domain)) ; else lg::level()(domain, false) | formatter()
// Like LOG_STREAM_NAMELESS except doesn't add newlines automatically
#define LOG_STREAM_NAMELESS_STREAMING(level, domain) if (lg::level().dont_log(domain)) ; else lg::level()(domain, false, false, true, true, false) | formatter()
// When using log_scope/log_scope2 it is nice to have all output indented.
#define LOG_STREAM_INDENT(level,domain) if (lg::level().dont_log(domain)) ; else lg::level()(domain, true, true) | formatter()
// If you have an explicit logger object and want to ignore the logging level, use this.
// Meant for cases where you explicitly call dont_log to avoid an expensive operation if the logging is disabled.
#define FORCE_LOG_TO(logger, domain) logger(domain) | formatter()
// always log (since it's at the error level) to the general log stream
// outputting the log domain and timestamp is disabled
// meant as a replacement to using cerr/cout, but that goes through the same logging infrastructure as everything else
#define PLAIN_LOG lg::err()(lg::general(), false, false, false, false, true) | formatter()
#define STREAMING_LOG lg::err()(lg::general(), false, false, false, false, false) | formatter()