2019-04-20 07:15:51 +00:00
|
|
|
/// The flogger: debug logging support for fish.
|
|
|
|
#ifndef FISH_FLOG_H
|
|
|
|
#define FISH_FLOG_H
|
|
|
|
|
|
|
|
#include "config.h" // IWYU pragma: keep
|
|
|
|
|
|
|
|
#include <stdio.h>
|
2019-10-13 22:50:48 +00:00
|
|
|
|
2019-04-20 07:15:51 +00:00
|
|
|
#include <string>
|
|
|
|
#include <type_traits>
|
|
|
|
#include <utility>
|
|
|
|
|
2019-10-13 22:50:48 +00:00
|
|
|
#include "global_safety.h"
|
2020-01-15 21:16:43 +00:00
|
|
|
#include "wcstringutil.h"
|
2019-10-13 22:50:48 +00:00
|
|
|
|
2019-04-20 07:15:51 +00:00
|
|
|
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-18 20:47:27 +00: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-18 20:49:59 +00:00
|
|
|
category_t exec_fork{L"exec-fork", L"Calls to fork()"};
|
|
|
|
|
2019-05-18 20:47:27 +00:00
|
|
|
category_t proc_job_run{L"proc-job-run", L"Jobs getting started or continued"};
|
2019-05-18 20:49:59 +00:00
|
|
|
|
2019-05-18 21:51:16 +00:00
|
|
|
category_t proc_termowner{L"proc-termowner", L"Terminal ownership events"};
|
2019-05-18 22:16:42 +00:00
|
|
|
|
2019-05-28 16:38:45 +00:00
|
|
|
category_t proc_internal_proc{L"proc-internal-proc", L"Internal (non-forked) process events"};
|
|
|
|
|
2019-06-13 21:29:13 +00:00
|
|
|
category_t proc_reap_internal{L"proc-reap-internal",
|
|
|
|
L"Reaping internal (non-forked) processes"};
|
|
|
|
|
|
|
|
category_t proc_reap_external{L"proc-reap-external", L"Reaping external (forked) processes"};
|
|
|
|
|
2019-05-18 22:16:42 +00:00
|
|
|
category_t env_locale{L"env-locale", L"Changes to locale variables"};
|
2019-05-29 21:20:07 +00:00
|
|
|
|
2019-06-09 20:21:58 +00:00
|
|
|
category_t env_export{L"env-export", L"Changes to exported variables"};
|
|
|
|
|
2019-05-29 21:20:07 +00:00
|
|
|
category_t topic_monitor{L"topic-monitor", L"Internal details of the topic monitor"};
|
2019-06-19 10:34:16 +00:00
|
|
|
category_t char_encoding{L"char-encoding", L"Character encoding issues"};
|
2019-08-15 20:42:23 +00:00
|
|
|
|
|
|
|
category_t history{L"history", L"Command history events"};
|
2019-08-18 18:14:45 +00:00
|
|
|
|
|
|
|
category_t profile_history{L"profile-history", L"History performance measurements"};
|
2019-11-23 21:37:15 +00:00
|
|
|
|
|
|
|
category_t iothread{L"iothread", L"Background IO thread events"};
|
2019-04-20 07:15:51 +00:00
|
|
|
};
|
|
|
|
|
|
|
|
/// 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, ...);
|
2019-10-19 01:08:22 +00:00
|
|
|
|
|
|
|
// Log outside of the usual flog usage.
|
|
|
|
void log_extra(const wchar_t *s) { log1(s); }
|
2019-04-20 07:15:51 +00:00
|
|
|
};
|
|
|
|
|
|
|
|
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();
|
|
|
|
|
2019-10-19 01:08:22 +00:00
|
|
|
/// Print some extra stuff to the flog file (stderr by default).
|
|
|
|
/// This is used by the tracing machinery.
|
|
|
|
void log_extra_to_flog_file(const wcstring &s);
|
|
|
|
|
2019-04-20 07:15:51 +00:00
|
|
|
/// 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
|