fish-shell/src/flog.h

146 lines
4.4 KiB
C
Raw Normal View History

/// The flogger: debug logging support for fish.
#ifndef FISH_FLOG_H
#define FISH_FLOG_H
#include "config.h" // IWYU pragma: keep
#include "global_safety.h"
#include <stdio.h>
#include <string>
#include <type_traits>
#include <utility>
using wcstring = std::wstring;
using wcstring_list_t = std::vector<wcstring>;
template <typename T>
class owning_lock;
namespace flog_details {
class category_list_t;
class category_t {
friend category_list_t;
category_t(const wchar_t *name, const wchar_t *desc, bool enabled = false);
public:
/// The name of this category.
const wchar_t *const name;
/// A (non-localized) description of the category.
const wchar_t *const description;
/// Whether the category is enabled.
relaxed_atomic_bool_t enabled;
};
class category_list_t {
category_list_t() = default;
public:
/// The singleton global category list instance.
static category_list_t *const g_instance;
/// What follows are the actual logging categories.
/// To add a new category simply define a new field, following the pattern.
category_t error{L"error", L"Serious unexpected errors (on by default)", true};
category_t debug{L"debug", L"Debugging aid (on by default)", true};
2019-05-19 04:47:27 +08:00
category_t exec_job_status{L"exec-job-status", L"Jobs changing status"};
category_t exec_job_exec{L"exec-job-exec", L"Jobs being executed"};
2019-05-19 04:49:59 +08:00
category_t exec_fork{L"exec-fork", L"Calls to fork()"};
2019-05-19 04:47:27 +08:00
category_t proc_job_run{L"proc-job-run", L"Jobs getting started or continued"};
2019-05-19 04:49:59 +08:00
category_t proc_termowner{L"proc-termowner", L"Terminal ownership events"};
category_t env_locale{L"env-locale", L"Changes to locale variables"};
};
/// The class responsible for logging.
/// This is protected by a lock.
class logger_t {
FILE *file_;
void log1(const wchar_t *);
void log1(const char *);
void log1(wchar_t);
void log1(char);
void log1(const wcstring &s) { log1(s.c_str()); }
void log1(const std::string &s) { log1(s.c_str()); }
template <typename T,
typename Enabler = typename std::enable_if<std::is_integral<T>::value>::type>
void log1(const T &v) {
log1(to_string(v));
}
template <typename T>
void log_args_impl(const T &arg) {
log1(arg);
}
template <typename T, typename... Ts>
void log_args_impl(const T &arg, const Ts &... rest) {
log1(arg);
log1(' ');
log_args_impl<Ts...>(rest...);
}
public:
void set_file(FILE *f) { file_ = f; }
logger_t();
template <typename... Args>
void log_args(const category_t &cat, const Args &... args) {
log1(cat.name);
log1(": ");
log_args_impl(args...);
log1('\n');
}
void log_fmt(const category_t &cat, const wchar_t *fmt, ...);
void log_fmt(const category_t &cat, const char *fmt, ...);
};
extern owning_lock<logger_t> g_logger;
} // namespace flog_details
/// Set the active flog categories according to the given wildcard \p wc.
void activate_flog_categories_by_pattern(const wcstring &wc);
/// Set the file that flog should output to.
/// flog does not close this file.
void set_flog_output_file(FILE *f);
/// \return a list of all categories, sorted by name.
std::vector<const flog_details::category_t *> get_flog_categories();
/// Output to the fish log a sequence of arguments, separated by spaces, and ending with a newline.
#define FLOG(wht, ...) \
do { \
if (flog_details::category_list_t::g_instance->wht.enabled) { \
flog_details::g_logger.acquire()->log_args( \
flog_details::category_list_t::g_instance->wht, __VA_ARGS__); \
} \
} while (0)
/// Output to the fish log a printf-style formatted string.
#define FLOGF(wht, ...) \
do { \
if (flog_details::category_list_t::g_instance->wht.enabled) { \
flog_details::g_logger.acquire()->log_fmt( \
flog_details::category_list_t::g_instance->wht, __VA_ARGS__); \
} \
} while (0)
#endif