GnuCash  5.6-93-g926250afab+
Macros | Typedefs | Enumerations | Functions


Logging and tracing facility. More...

Macros

#define QOF_MOD_ENGINE   "qof.engine"
 
#define PRETTY_FUNC_NAME   qof_log_prettify(G_STRFUNC)
 
#define FATAL(format, args...)
 Log a fatal error. More...
 
#define PERR(format, args...)
 Log a serious error. More...
 
#define PWARN(format, args...)
 Log a warning. More...
 
#define PINFO(format, args...)
 Print an informational note. More...
 
#define DEBUG(format, args...)
 Print a debugging message. More...
 
#define ENTER(format, args...)
 Print a function entry debugging message. More...
 
#define LEAVE(format, args...)
 Print a function exit debugging message. More...
 
#define gnc_leave_return_val_if_fail(test, val)
 Replacement for g_return_val_if_fail, but calls LEAVE if the test fails. More...
 
#define gnc_leave_return_if_fail(test)
 Replacement for g_return_if_fail, but calls LEAVE if the test fails. More...
 

Typedefs

typedef const gchar * QofLogModule
 

Enumerations

enum  QofLogLevel {
  QOF_LOG_FATAL = G_LOG_LEVEL_ERROR, QOF_LOG_ERROR = G_LOG_LEVEL_CRITICAL, QOF_LOG_WARNING = G_LOG_LEVEL_WARNING, QOF_LOG_MESSAGE = G_LOG_LEVEL_MESSAGE,
  QOF_LOG_INFO = G_LOG_LEVEL_INFO, QOF_LOG_DEBUG = G_LOG_LEVEL_DEBUG
}
 

Functions

const char * qof_log_level_to_string (QofLogLevel lvl)
 
QofLogLevel qof_log_level_from_string (const char *str)
 
void qof_log_indent (void)
 Indents one level; see ENTER macro. More...
 
void qof_log_dedent (void)
 De-dent one level, capped at 0; see LEAVE macro.
 
void qof_log_init (void)
 Initialize the error logging subsystem. More...
 
void qof_log_set_level (QofLogModule module, QofLogLevel level)
 Set the logging level of the given log_module. More...
 
void qof_log_set_file (FILE *outfile)
 Specify an alternate log output, to pipe or file. More...
 
void qof_log_init_filename (const gchar *logfilename)
 Specify a filename for log output. More...
 
void qof_log_init_filename_special (const char *log_to_filename)
 If log_to_filename is "stderr" or "stdout" (exactly, case-insensitive), then those special files are used; otherwise, the literal filename as given, as qof_log_init_filename(gchar*)
 
void qof_log_parse_log_config (const char *filename)
 
Parse a log-configuration file. More...
 
void qof_log_shutdown (void)
 Be nice, close the logfile if possible. More...
 
const gchar * qof_log_prettify (const gchar *name)
 Cleans up subroutine names. More...
 
gboolean qof_log_check (QofLogModule log_module, QofLogLevel log_level)
 Check to see if the given log_module is configured to log at the given log_level. More...
 

Detailed Description


Logging and tracing facility.

See also
"Logging overhaul" announcement https://lists.gnucash.org/pipermail/gnucash-devel/2007-February/019836.html

qof_log_init(void) installs a handler that interprets the "log_domain" as a "."-separated path. Log level thresholds can be set for each level in the tree. When a message is logged, the longest level match is found, and used as the threshold.

For instance, we can set the levels as such:

  "qof"                        = WARN
  "gnc"                        = WARN
  "gnc.ui"                     = INFO
  "gnc.ui.plugin-page.sx-list" = DEBUG

When code in the log_module of "gnc.import" attempts to log at DEBUG (let's say), the handler will attempt to match the log domain to successively-longer paths: first "", then "gnc", then "gnc.import". Given the settings above, the path "gnc" will match – at a level of "WARN" – and the DEBUG-level log will be rejected. When code in the log domain of "gnc.ui.plugin-page.sx-list" logs at DEBUG, however, it will match at DEBUG, and be allowed.

The current log format is as above:

    * [timestamp] [level] <[log-domain]> [message]

The timestamp and level are constant width (level is 5 characters). The log domain is re-iterated, which gives some context, but could become annoying if they get long.

Trailing newlines (e.g. PINFO("...\n", ...)) are removed; the logger will newline separate output.

Best Practices

Code should:

See also
qof_log_parse_log_config(const char*)

Macro Definition Documentation

◆ DEBUG

#define DEBUG (   format,
  args... 
)
Value:
do { \
if (qof_log_check(log_module, QOF_LOG_DEBUG)) { \
g_log (log_module, G_LOG_LEVEL_DEBUG, \
"[%s] " format, PRETTY_FUNC_NAME , ## args); \
} \
} while(0)
gboolean qof_log_check(QofLogModule log_module, QofLogLevel log_level)
Check to see if the given log_module is configured to log at the given log_level. ...
Definition: qoflog.cpp:324

Print a debugging message.

Definition at line 264 of file qoflog.h.

◆ ENTER

#define ENTER (   format,
  args... 
)
Value:
do { \
if (qof_log_check(log_module, QOF_LOG_DEBUG)) { \
g_log (log_module, G_LOG_LEVEL_DEBUG, \
"[enter %s:%s()] " format, __FILE__, \
PRETTY_FUNC_NAME , ## args); \
qof_log_indent(); \
} \
} while (0)
gboolean qof_log_check(QofLogModule log_module, QofLogLevel log_level)
Check to see if the given log_module is configured to log at the given log_level. ...
Definition: qoflog.cpp:324

Print a function entry debugging message.

Definition at line 272 of file qoflog.h.

◆ FATAL

#define FATAL (   format,
  args... 
)
Value:
do { \
g_log (log_module, G_LOG_LEVEL_ERROR, \
"[%s()] " format, PRETTY_FUNC_NAME , ## args); \
} while (0)

Log a fatal error.

Definition at line 238 of file qoflog.h.

◆ gnc_leave_return_if_fail

#define gnc_leave_return_if_fail (   test)
Value:
do { \
if (! (test)) { LEAVE(""); } \
g_return_if_fail(test); \
} while (0);
#define LEAVE(format, args...)
Print a function exit debugging message.
Definition: qoflog.h:282

Replacement for g_return_if_fail, but calls LEAVE if the test fails.

Definition at line 300 of file qoflog.h.

◆ gnc_leave_return_val_if_fail

#define gnc_leave_return_val_if_fail (   test,
  val 
)
Value:
do { \
if (! (test)) { LEAVE(""); } \
g_return_val_if_fail(test, val); \
} while (0);
#define LEAVE(format, args...)
Print a function exit debugging message.
Definition: qoflog.h:282

Replacement for g_return_val_if_fail, but calls LEAVE if the test fails.

Definition at line 294 of file qoflog.h.

◆ LEAVE

#define LEAVE (   format,
  args... 
)
Value:
do { \
if (qof_log_check(log_module, QOF_LOG_DEBUG)) { \
qof_log_dedent(); \
g_log (log_module, G_LOG_LEVEL_DEBUG, \
"[leave %s()] " format, \
PRETTY_FUNC_NAME , ## args); \
} \
} while (0)
gboolean qof_log_check(QofLogModule log_module, QofLogLevel log_level)
Check to see if the given log_module is configured to log at the given log_level. ...
Definition: qoflog.cpp:324

Print a function exit debugging message.

Definition at line 282 of file qoflog.h.

◆ PERR

#define PERR (   format,
  args... 
)
Value:
do { \
g_log (log_module, G_LOG_LEVEL_CRITICAL, \
"[%s()] " format, PRETTY_FUNC_NAME , ## args); \
} while (0)

Log a serious error.

Definition at line 244 of file qoflog.h.

◆ PINFO

#define PINFO (   format,
  args... 
)
Value:
do { \
if (qof_log_check(log_module, QOF_LOG_INFO)) { \
g_log (log_module, G_LOG_LEVEL_INFO, \
"[%s] " format, PRETTY_FUNC_NAME , ## args); \
} \
} while (0)
gboolean qof_log_check(QofLogModule log_module, QofLogLevel log_level)
Check to see if the given log_module is configured to log at the given log_level. ...
Definition: qoflog.cpp:324

Print an informational note.

Definition at line 256 of file qoflog.h.

◆ PWARN

#define PWARN (   format,
  args... 
)
Value:
do { \
g_log (log_module, G_LOG_LEVEL_WARNING, \
"[%s()] " format, PRETTY_FUNC_NAME , ## args); \
} while (0)

Log a warning.

Definition at line 250 of file qoflog.h.

Function Documentation

◆ qof_log_check()

gboolean qof_log_check ( QofLogModule  log_module,
QofLogLevel  log_level 
)

Check to see if the given log_module is configured to log at the given log_level.

This implements the "log.path.hierarchy" logic.

Definition at line 324 of file qoflog.cpp.

325 {
326 // Check the global levels
327  if (level > current_max)
328  return FALSE;
329  if (level <= default_level)
330  return TRUE;
331  auto module = get_modules();
332  // If the level <= the default then no need to look further.
333  if (level <= module->m_level)
334  return TRUE;
335 
336  if (!domain)
337  return FALSE;
338 
339  auto domain_vec = split_domain(domain);
340 
341  for (const auto& part : domain_vec)
342  {
343  auto iter = std::find_if(module->m_children.begin(),
344  module->m_children.end(),
345  [part](auto& child) {
346  return child && part == child->m_name; });
347 
348  if (iter == module->m_children.end())
349  return FALSE;
350 
351  if (level <= (*iter)->m_level)
352  return TRUE;
353 
354  module = iter->get();
355  }
356  return FALSE;
357 }

◆ qof_log_indent()

void qof_log_indent ( void  )

Indents one level; see ENTER macro.

Definition at line 130 of file qoflog.cpp.

131 {
132  qof_log_num_spaces += QOF_LOG_INDENT_WIDTH;
133 }

◆ qof_log_init()

void qof_log_init ( void  )

Initialize the error logging subsystem.

Defaults to a level-threshold of "warning", and logging to stderr.

Definition at line 156 of file qoflog.cpp.

157 {
158  qof_log_init_filename(nullptr);
159 }
void qof_log_init_filename(const gchar *log_filename)
Specify a filename for log output.
Definition: qoflog.cpp:207

◆ qof_log_init_filename()

void qof_log_init_filename ( const gchar *  logfilename)

Specify a filename for log output.

Definition at line 207 of file qoflog.cpp.

208 {
209  gboolean warn_about_missing_permission = FALSE;
210  auto modules = get_modules();
211 
212  if (!qof_logger_format)
213  qof_logger_format = g_strdup ("* %s %*s <%s> %*s%s%s"); //default format
214 
215  if (log_filename)
216  {
217  int fd;
218  gchar *fname;
219 
220  if (fout != nullptr && fout != stderr && fout != stdout)
221  fclose(fout);
222 
223  fname = g_strconcat(log_filename, ".XXXXXX.log", nullptr);
224 
225  if ((fd = g_mkstemp(fname)) != -1)
226  {
227 #if PLATFORM(WINDOWS)
228  /* MSVC compiler: Somehow the OS thinks file descriptor from above
229  * still isn't open. So we open normally with the file name and that's it. */
230  fout = g_fopen(fname, "wb");
231 #else
232  /* We must not overwrite /dev/null */
233  g_assert(g_strcmp0(log_filename, "/dev/null") != 0);
234 
235  /* Windows prevents renaming of open files, so the next command silently fails there
236  * No problem, the filename on Windows will simply have the random characters */
237  g_rename(fname, log_filename);
238  fout = fdopen(fd, "w");
239 #endif
240  if (!fout)
241  warn_about_missing_permission = TRUE;
242  }
243  else
244  {
245  warn_about_missing_permission = TRUE;
246  fout = stderr;
247  }
248  g_free(fname);
249  }
250 
251  if (!fout)
252  fout = stderr;
253 
254  if (previous_handler == nullptr)
255  previous_handler = g_log_set_default_handler(log4glib_handler, modules);
256 
257  if (warn_about_missing_permission)
258  {
259  g_critical("Cannot open log output file \"%s\", using stderr.", log_filename);
260  }
261 }

◆ qof_log_parse_log_config()

void qof_log_parse_log_config ( const char *  filename)


Parse a log-configuration file.

A GKeyFile-format file of the schema:

   [levels]
   # log.ger.path=level
   gnc.engine.sx=debug
   gnc.gui.sx=debug
   gnc.import-export.qif.parse=debug
   [output]
   # to=["stderr"|"stdout"|filename]
   to=stderr

Definition at line 415 of file qoflog.cpp.

416 {
417  const gchar *levels_group = "levels", *output_group = "output";
418  GError *err = nullptr;
419  GKeyFile *conf = g_key_file_new();
420 
421  if (!g_key_file_load_from_file(conf, filename, G_KEY_FILE_NONE, &err))
422  {
423  g_warning("unable to parse [%s]: %s", filename, err->message);
424  g_error_free(err);
425  return;
426  }
427 
428  DEBUG("parsing log config from [%s]", filename);
429  if (g_key_file_has_group(conf, levels_group))
430  {
431  gsize num_levels;
432  unsigned int key_idx;
433  gchar **levels;
434  gint logger_max_name_length = 12;
435  gchar *str = nullptr;
436 
437  levels = g_key_file_get_keys(conf, levels_group, &num_levels, nullptr);
438 
439  for (key_idx = 0; key_idx < num_levels && levels[key_idx] != nullptr; key_idx++)
440  {
441  QofLogLevel level;
442  gchar *logger_name = nullptr, *level_str = nullptr;
443 
444  logger_name = g_strdup(levels[key_idx]);
445  logger_max_name_length = MAX (logger_max_name_length, (gint) strlen (logger_name));
446  level_str = g_key_file_get_string(conf, levels_group, logger_name, nullptr);
447  level = qof_log_level_from_string(level_str);
448 
449  DEBUG("setting log [%s] to level [%s=%d]", logger_name, level_str, level);
450  qof_log_set_level(logger_name, level);
451 
452  g_free(logger_name);
453  g_free(level_str);
454  }
455 
456  str = g_strdup_printf ("%d", logger_max_name_length);
457  if (qof_logger_format)
458  g_free (qof_logger_format);
459  qof_logger_format = g_strconcat ("* %s %*s <%-", str, ".", str, "s> %*s%s%s", nullptr);
460 
461  g_free (str);
462  g_strfreev(levels);
463  }
464 
465  if (g_key_file_has_group(conf, output_group))
466  {
467  gsize num_outputs;
468  unsigned int output_idx;
469  gchar **outputs;
470 
471  outputs = g_key_file_get_keys(conf, output_group, &num_outputs, nullptr);
472  for (output_idx = 0; output_idx < num_outputs && outputs[output_idx] != nullptr; output_idx++)
473  {
474  gchar *key = outputs[output_idx];
475  gchar *value;
476 
477  if (g_ascii_strcasecmp("to", key) != 0)
478  {
479  g_warning("unknown key [%s] in [outputs], skipping", key);
480  continue;
481  }
482 
483  value = g_key_file_get_string(conf, output_group, key, nullptr);
484  DEBUG("setting [output].to=[%s]", value);
486  g_free(value);
487  }
488  g_strfreev(outputs);
489  }
490 
491  g_key_file_free(conf);
492 }
void qof_log_set_level(QofLogModule log_module, QofLogLevel level)
Set the logging level of the given log_module.
Definition: qoflog.cpp:291
#define DEBUG(format, args...)
Print a debugging message.
Definition: qoflog.h:264
void qof_log_init_filename_special(const char *log_to_filename)
If log_to_filename is "stderr" or "stdout" (exactly, case-insensitive), then those special files are ...
Definition: qoflog.cpp:396

◆ qof_log_prettify()

const gchar* qof_log_prettify ( const gchar *  name)

Cleans up subroutine names.

AIX/xlC has the habit of printing signatures not names; clean this up. On other operating systems, truncate name to QOF_LOG_MAX_CHARS chars.

◆ qof_log_set_file()

void qof_log_set_file ( FILE *  outfile)

Specify an alternate log output, to pipe or file.

Definition at line 145 of file qoflog.cpp.

146 {
147  if (!outfile)
148  {
149  fout = stderr;
150  return;
151  }
152  fout = outfile;
153 }

◆ qof_log_set_level()

void qof_log_set_level ( QofLogModule  module,
QofLogLevel  level 
)

Set the logging level of the given log_module.

Definition at line 291 of file qoflog.cpp.

292 {
293  if (!log_module || level == QOF_LOG_FATAL)
294  return;
295 
296  if (level > current_max)
297  current_max = level;
298 
299  auto module_parts = split_domain(log_module);
300  auto module = get_modules();
301  for (auto part : module_parts)
302  {
303  auto iter = std::find_if(module->m_children.begin(),
304  module->m_children.end(),
305  [part](auto& child){
306  return child && part == child->m_name;
307  });
308  if (iter == module->m_children.end())
309  {
310  auto child = std::make_unique<ModuleEntry>(part, default_level);
311  module->m_children.emplace_back(std::move(child));
312  module = module->m_children.back().get();
313  }
314  else
315  {
316  module = iter->get();
317  }
318  }
319  module->m_level = level;
320 }

◆ qof_log_shutdown()

void qof_log_shutdown ( void  )

Be nice, close the logfile if possible.

Definition at line 264 of file qoflog.cpp.

265 {
266  if (fout && fout != stderr && fout != stdout)
267  {
268  fclose(fout);
269  fout = nullptr;
270  }
271 
272  if (function_buffer)
273  {
274  g_free(function_buffer);
275  function_buffer = nullptr;
276  }
277 
278  if (_modules != nullptr)
279  {
280  _modules = nullptr;
281  }
282 
283  if (previous_handler != nullptr)
284  {
285  g_log_set_default_handler(previous_handler, nullptr);
286  previous_handler = nullptr;
287  }
288 }