The Battle for Wesnoth  1.19.10+dev
log.cpp
Go to the documentation of this file.
1 /*
2  Copyright (C) 2004 - 2025
3  by Guillaume Melquiond <guillaume.melquiond@gmail.com>
4  Copyright (C) 2003 by David White <dave@whitevine.net>
5  Part of the Battle for Wesnoth Project https://www.wesnoth.org/
6 
7  This program is free software; you can redistribute it and/or modify
8  it under the terms of the GNU General Public License as published by
9  the Free Software Foundation; either version 2 of the License, or
10  (at your option) any later version.
11  This program is distributed in the hope that it will be useful,
12  but WITHOUT ANY WARRANTY.
13 
14  See the COPYING file for more details.
15 */
16 
17 /**
18  * @file
19  * Standard logging facilities (implementation).
20  * See also the command line switches --logdomains and --log-@<level@>="domain".
21  */
22 
23 #include "log.hpp"
24 #include "filesystem.hpp"
25 #include "mt_rng.hpp"
26 #include "serialization/chrono.hpp"
28 #include "utils/general.hpp"
29 
30 #include <boost/algorithm/string.hpp>
31 
32 #include <map>
33 #include <mutex>
34 #include <iostream>
35 #include <iomanip>
36 
37 #ifdef _WIN32
38 #include <io.h>
39 #endif
40 
41 static lg::log_domain log_setup("logsetup");
42 #define ERR_LS LOG_STREAM(err, log_setup)
43 #define WRN_LS LOG_STREAM(warn, log_setup)
44 #define LOG_LS LOG_STREAM(info, log_setup)
45 #define DBG_LS LOG_STREAM(debug, log_setup)
46 
47 namespace {
48 
49 class null_streambuf : public std::streambuf
50 {
51  virtual int overflow(int c) { return std::char_traits< char >::not_eof(c); }
52 public:
53  null_streambuf() {}
54 };
55 
56 } // end anonymous namespace
57 
58 static std::ostream null_ostream(new null_streambuf);
59 static int indent = 0;
60 static bool timestamp = true;
61 static bool precise_timestamp = false;
62 static std::mutex log_mutex;
63 
64 static bool log_sanitization = true;
65 
66 /** whether the current logs directory is writable */
67 static utils::optional<bool> is_log_dir_writable_ = utils::nullopt;
68 /** alternative stream to write data to */
69 static std::ostream *output_stream_ = nullptr;
70 
71 /**
72  * @return std::cerr if the redirect_output_setter isn't being used, output_stream_ if it is
73  */
74 static std::ostream& output()
75 {
76  if(output_stream_) {
77  return *output_stream_;
78  }
79  return std::cerr;
80 }
81 
82 /** path to the current log file; does not include the extension */
83 static std::string output_file_path_ = "";
84 /** path to the current logs directory; may change after being initially set if a custom userdata directory is given on the command line */
85 static std::string logs_dir_ = "";
86 
87 namespace lg {
88 
89 std::ostringstream& operator<<(std::ostringstream& oss, const lg::severity severity)
90 {
91  oss << static_cast<int>(severity);
92  return oss;
93 }
94 
95 bool is_not_log_file(const std::string& fn)
96 {
97  return !(boost::algorithm::istarts_with(fn, lg::log_file_prefix) &&
98  boost::algorithm::iends_with(fn, lg::log_file_suffix));
99 }
100 
101 void rotate_logs(const std::string& log_dir)
102 {
103  // if logging to file is disabled, don't rotate the logs
104  if(output_file_path_.empty()) {
105  return;
106  }
107 
108  std::vector<std::string> files;
109  filesystem::get_files_in_dir(log_dir, &files);
110 
112 
113  if(files.size() <= lg::max_logs) {
114  return;
115  }
116 
117  // Sorting the file list and deleting all but the last max_logs items
118  // should hopefully be faster than stat'ing every single file for its
119  // time attributes (which aren't very reliable to begin with).
120 
121  std::sort(files.begin(), files.end());
122 
123  for(std::size_t j = 0; j < files.size() - lg::max_logs; ++j) {
124  const std::string path = log_dir + '/' + files[j];
125  LOG_LS << "rotate_logs(): delete " << path;
127  ERR_LS << "rotate_logs(): failed to delete " << path << "!";
128  }
129  }
130 }
131 
132 std::string unique_log_filename()
133 {
134  std::ostringstream o;
135  const auto now = std::chrono::system_clock::now();
136  randomness::mt_rng rng;
137 
139  << chrono::format_local_timestamp(now, "%Y%m%d-%H%M%S-")
140  << rng.get_next_random();
141 
142  return o.str();
143 }
144 
146 {
147  std::string dummy_log = filesystem::get_logs_dir()+"/dummy.log";
148 
149  // log directory doesn't exist and can't be created
151  is_log_dir_writable_ = false;
152  return;
153  }
154 
155  // can't create and write new log files
156  try {
157  filesystem::write_file(dummy_log, " ");
158  } catch(const filesystem::io_exception&) {
159  is_log_dir_writable_ = false;
160  return;
161  }
162 
163  // confirm that file exists and was written to
164  if(filesystem::file_size(dummy_log) != 1) {
165  is_log_dir_writable_ = false;
166  }
167 
168  // can't delete files - prevents log rotation
169  if(filesystem::file_exists(dummy_log) && !filesystem::delete_file(dummy_log)) {
170  is_log_dir_writable_ = false;
171  return;
172  }
173 
174  is_log_dir_writable_ = true;
175 }
176 
178 {
179  if(logs_dir_ == filesystem::get_logs_dir() || logs_dir_ == "") {
180  return;
181  }
182 
184 
185  if(is_log_dir_writable_.value_or(false)) {
186 #ifdef _WIN32
187  std::string old_path = output_file_path_;
189 
190  // flush and close existing log files, since Windows doesn't allow moving open files
191  std::fflush(stderr);
192  std::cerr.flush();
193  if(!std::freopen("NUL", "a", stderr)) {
194  std::cerr << "Failed to close stderr log file: '" << old_path << "'";
195  // stderr is where basically all output goes through, so if that fails then don't attempt anything else
196  // moving just the stdout log would be pointless
197  return;
198  }
199  std::fflush(stdout);
200  std::cout.flush();
201  if(!std::freopen("NUL", "a", stdout)) {
202  std::cerr << "Failed to close stdout log file: '" << old_path << "'";
203  }
204 
205  // move the .log and .out.log files
206  // stdout and stderr are set to NUL currently so nowhere to send info on failure
207  if(rename((old_path+lg::log_file_suffix).c_str(), (output_file_path_+lg::log_file_suffix).c_str()) == -1) {
208  return;
209  }
210  rename((old_path+lg::out_log_file_suffix).c_str(), (output_file_path_+lg::out_log_file_suffix).c_str());
211 
212  // reopen to log files at new location
213  // stdout and stderr are still NUL if freopen fails, so again nowhere to send info on failure
214  std::fflush(stderr);
215  std::cerr.flush();
216  std::freopen((output_file_path_+lg::log_file_suffix).c_str(), "a", stderr);
217 
218  std::fflush(stdout);
219  std::cout.flush();
220  std::freopen((output_file_path_+lg::out_log_file_suffix).c_str(), "a", stdout);
221 #else
222  std::string old_path = get_log_file_path();
224 
225  // non-Windows can just move the file
226  if(rename(old_path.c_str(), get_log_file_path().c_str()) == -1) {
227  std::cerr << "Failed to rename log file from '" << old_path << "' to '" << output_file_path_ << "'";
228  }
229 #endif
230  }
231 }
232 
234 {
236  // if the log directory is not writable, then don't try to do anything.
237  // if the log directory is writable, then setup logging and rotate the logs.
238  // if the optional isn't set, then logging to file has been disabled, so don't try to do anything
239  if(is_log_dir_writable_.value_or(false)) {
240  // get the log file stream and assign cerr+cout to it
243 
244  // IMPORTANT: apparently redirecting stderr/stdout will also redirect std::cerr/std::cout, but the reverse is not true
245  // redirecting std::cerr/std::cout will *not* redirect stderr/stdout
246 
247  // redirect stderr to file
248  std::fflush(stderr);
249  std::cerr.flush();
250  if(!std::freopen((output_file_path_+lg::log_file_suffix).c_str(), "w", stderr)) {
251  std::cerr << "Failed to redirect stderr to a file!";
252  }
253 
254  // redirect stdout to file
255  // separate handling for Windows since dup2() just... doesn't work for GUI apps there apparently
256  // redirect to a separate file on Windows as well, since otherwise two streams independently writing to the same file can cause weirdness
257 #ifdef _WIN32
258  std::fflush(stdout);
259  std::cout.flush();
260  if(!std::freopen((output_file_path_+lg::out_log_file_suffix).c_str(), "w", stdout)) {
261  std::cerr << "Failed to redirect stdout to a file!";
262  }
263 #else
264  if(dup2(STDERR_FILENO, STDOUT_FILENO) == -1) {
265  std::cerr << "Failed to redirect stdout to a file!";
266  }
267 #endif
268 
269  // make stdout unbuffered - otherwise some output might be lost
270  // in practice shouldn't make much difference either way, given how little output goes through stdout/std::cout
271  if(setvbuf(stdout, nullptr, _IONBF, 2) == -1) {
272  std::cerr << "Failed to set stdout to be unbuffered";
273  }
274 
276  }
277 }
278 
279 utils::optional<bool> log_dir_writable()
280 {
281  return is_log_dir_writable_;
282 }
283 
284 std::string get_log_file_path()
285 {
287 }
288 
290  : old_stream_(output_stream_)
291 {
292  output_stream_ = &stream;
293 }
294 
296 {
298 }
299 
300 typedef std::map<std::string, severity> domain_map;
303 void timestamps(bool t) { timestamp = t; }
304 void precise_timestamps(bool pt) { precise_timestamp = pt; }
305 
307 {
308  static logger lg("error", severity::LG_ERROR);
309  return lg;
310 }
311 
313 {
314  static logger lg("warning", severity::LG_WARN);
315  return lg;
316 }
317 
319 {
320  static logger lg("info", severity::LG_INFO);
321  return lg;
322 }
323 
325 {
326  static logger lg("debug", severity::LG_DEBUG);
327  return lg;
328 }
329 
330 static log_domain dom("general");
331 
333 {
334  return dom;
335 }
336 
338  : domain_(nullptr)
339 {
340  // Indirection to prevent initialization depending on link order.
341  if (!domains) domains = new domain_map;
342  domain_ = &*domains->insert(logd(name, severity)).first;
343  domain_->second = severity;
344 }
345 
346 bool set_log_domain_severity(const std::string& name, severity severity)
347 {
348  if (name == "all") {
349  for(logd &l : *domains) {
350  l.second = severity;
351  }
352  return true;
353  } else {
354  bool any_matched = false;
355  for (logd &l : *domains) {
356  if (utils::wildcard_string_match(l.first, name)) {
357  l.second = severity;
358  any_matched = true;
359  }
360  }
361  return any_matched;
362  }
363 }
364 bool set_log_domain_severity(const std::string& name, const logger &lg) {
365  return set_log_domain_severity(name, lg.get_severity());
366 }
367 
368 bool get_log_domain_severity(const std::string& name, severity &severity)
369 {
370  domain_map::iterator it = domains->find(name);
371  if (it == domains->end())
372  return false;
373  severity = it->second;
374  return true;
375 }
376 
377 std::string list_log_domains(const std::string& filter)
378 {
379  std::ostringstream res;
380  for(logd &l : *domains) {
381  if(l.first.find(filter) != std::string::npos)
382  res << l.first << "\n";
383  }
384  return res.str();
385 }
386 
389 }
390 
392  set_strict_severity(lg.get_severity());
393 }
394 
395 static bool strict_threw_ = false;
396 
397 bool broke_strict() {
398  return strict_threw_;
399 }
400 
401 void set_log_sanitize(bool sanitize) {
402  log_sanitization = sanitize;
403 }
404 
405 std::string sanitize_log(const std::string& logstr)
406 {
407  if(!log_sanitization) {
408  return logstr;
409  }
410 
411  std::string str = logstr;
412 
413 #ifdef _WIN32
414  const char* user_name = getenv("USERNAME");
415 #else
416  const char* user_name = getenv("USER");
417 #endif
418 
419  if(user_name != nullptr) {
420  boost::replace_all(str, std::string("/") + user_name + "/", "/USER/");
421  boost::replace_all(str, std::string("\\") + user_name + "\\", "\\USER\\");
422  }
423 
424  return str;
425 }
426 
428  const log_domain& domain,
429  bool show_names,
430  bool do_indent,
431  bool show_timestamps,
432  bool break_strict,
433  bool auto_newline) const
434 {
435  if (severity_ > domain.domain_->second) {
436  return null_ostream;
437  } else {
438  log_in_progress stream = output();
439  if(do_indent) {
440  stream.set_indent(indent);
441  }
442  if (timestamp && show_timestamps) {
443  stream.enable_timestamp();
444  }
445  if (show_names) {
446  stream.set_prefix(formatter() << name_ << ' ' << domain.domain_->first << ": ");
447  }
448  if (!strict_threw_ && severity_ <= strict_level_ && break_strict) {
449  stream | formatter() << "Error (strict mode, strict_level = " << strict_level_ << "): wesnoth reported on channel " << name_ << " " << domain.domain_->first << std::endl;
450  strict_threw_ = true;
451  }
452  stream.set_auto_newline(auto_newline);
453  return stream;
454  }
455 }
456 
457 log_in_progress::log_in_progress(std::ostream& stream)
458  : stream_(stream)
459 {}
460 
462 {
463  std::scoped_lock lock(log_mutex);
464  for(int i = 0; i < indent; ++i)
465  stream_ << " ";
466  if(timestamp_) {
467  auto now = std::chrono::system_clock::now();
468  stream_ << chrono::format_local_timestamp(now); // Truncates precision to seconds
469  if(precise_timestamp) {
470  auto as_seconds = std::chrono::time_point_cast<std::chrono::seconds>(now);
471  auto fractional = std::chrono::duration_cast<std::chrono::microseconds>(now - as_seconds);
472  stream_ << "." << std::setw(6) << fractional.count();
473  }
474  stream_ << " ";
475  }
476  stream_ << prefix_ << sanitize_log(message.str());
477  if(auto_newline_) {
478  stream_ << std::endl;
479  }
480 }
481 
483  indent_ = level;
484 }
485 
487  timestamp_ = true;
488 }
489 
490 void log_in_progress::set_prefix(const std::string& prefix) {
491  prefix_ = prefix;
492 }
493 
494 void log_in_progress::set_auto_newline(bool auto_newline) {
495  auto_newline_ = auto_newline;
496 }
497 
498 void scope_logger::do_log_entry(const std::string& str) noexcept
499 {
500  str_ = str;
501  start_ = std::chrono::steady_clock::now();
502  debug()(domain_, false, true) | formatter() << "{ BEGIN: " << str_;
503  ++indent;
504 }
505 
507 {
508  --indent;
509  auto output = debug()(domain_, false, true);
510  output.set_indent(indent);
511  if(timestamp) output.enable_timestamp();
512  auto now = std::chrono::steady_clock::now();
513  auto elapsed = std::chrono::duration_cast<std::chrono::microseconds>(now - start_);
514  output | formatter() << "} END: " << str_ << " (took " << elapsed.count() << "us)"; // FIXME c++20 stream: operator
515 }
516 
517 std::stringstream& log_to_chat()
518 {
519  static std::stringstream lg;
520  return lg;
521 }
522 
523 } // end namespace lg
double t
Definition: astarsearch.cpp:63
std::ostringstream wrapper.
Definition: formatter.hpp:40
logd * domain_
Definition: log.hpp:125
log_domain(char const *name, severity severity=severity::LG_WARN)
Definition: log.cpp:337
std::string prefix_
Definition: log.hpp:195
std::ostream & stream_
Definition: log.hpp:192
void set_auto_newline(bool enabled)
Definition: log.cpp:494
void set_prefix(const std::string &prefix)
Definition: log.cpp:490
void enable_timestamp()
Definition: log.cpp:486
void operator|(const formatter &message)
Definition: log.cpp:461
bool auto_newline_
Definition: log.hpp:196
void set_indent(int level)
Definition: log.cpp:482
log_in_progress(std::ostream &stream)
Definition: log.cpp:457
char const * name_
Definition: log.hpp:207
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
Definition: log.cpp:427
severity severity_
Definition: log.hpp:208
std::ostream * old_stream_
The previously set redirection.
Definition: log.hpp:117
redirect_output_setter(std::ostream &stream)
Constructor.
Definition: log.cpp:289
const log_domain & domain_
Definition: log.hpp:241
void do_log_entry(const std::string &str) noexcept
Definition: log.cpp:498
std::string str_
Definition: log.hpp:242
std::chrono::steady_clock::time_point start_
Definition: log.hpp:240
void do_log_exit() noexcept
Definition: log.cpp:506
uint32_t get_next_random()
Get a new random number.
Definition: mt_rng.cpp:63
Declarations for File-IO.
std::size_t i
Definition: function.cpp:1022
#define ERR_LS
Definition: log.cpp:42
static bool precise_timestamp
Definition: log.cpp:61
static std::string logs_dir_
path to the current logs directory; may change after being initially set if a custom userdata directo...
Definition: log.cpp:85
static utils::optional< bool > is_log_dir_writable_
whether the current logs directory is writable
Definition: log.cpp:67
#define LOG_LS
Definition: log.cpp:44
static std::string output_file_path_
path to the current log file; does not include the extension
Definition: log.cpp:83
static std::ostream null_ostream(new null_streambuf)
static std::mutex log_mutex
Definition: log.cpp:62
static int indent
Definition: log.cpp:59
static std::ostream & output()
Definition: log.cpp:74
static bool log_sanitization
Definition: log.cpp:64
static std::ostream * output_stream_
alternative stream to write data to
Definition: log.cpp:69
static lg::log_domain log_setup("logsetup")
static bool timestamp
Definition: log.cpp:60
Standard logging facilities (interface).
auto format_local_timestamp(const std::chrono::system_clock::time_point &time, std::string_view format="%F %T")
Definition: chrono.hpp:62
void get_files_in_dir(const std::string &dir, std::vector< std::string > *files, std::vector< std::string > *dirs, name_mode mode, filter_mode filter, reorder_mode reorder, file_tree_checksum *checksum)
Get a list of all files and/or directories in a given directory.
Definition: filesystem.cpp:450
bool delete_file(const std::string &filename)
static bool file_exists(const bfs::path &fpath)
Definition: filesystem.cpp:328
int file_size(const std::string &fname)
Returns the size of a file, or -1 if the file doesn't exist.
void write_file(const std::string &fname, const std::string &data, std::ios_base::openmode mode)
Throws io_exception if an error occurs.
std::string get_logs_dir()
Definition: filesystem.cpp:861
bool make_directory(const std::string &dirname)
std::string path
Definition: filesystem.cpp:93
Definition: pump.hpp:41
bool get_log_domain_severity(const std::string &name, severity &severity)
Definition: log.cpp:368
logger & err()
Definition: log.cpp:306
severity
Definition: log.hpp:82
std::string list_log_domains(const std::string &filter)
Definition: log.cpp:377
log_domain & general()
Definition: log.cpp:332
bool broke_strict()
Definition: log.cpp:397
logger & debug()
Definition: log.cpp:324
const std::string out_log_file_suffix
Definition: log.hpp:70
void rotate_logs(const std::string &log_dir)
Check how many log files exist and delete the oldest when there's too many.
Definition: log.cpp:101
std::string unique_log_filename()
Generate a unique file name using the current timestamp and a randomly generated number.
Definition: log.cpp:132
void set_log_to_file()
Do the initial redirection to a log file if the logs directory is writable.
Definition: log.cpp:233
static log_domain dom("general")
void set_log_sanitize(bool sanitize)
toggle log sanitization
Definition: log.cpp:401
utils::optional< bool > log_dir_writable()
Returns the result set by check_log_dir_writable().
Definition: log.cpp:279
void move_log_file()
Move the log file to another directory.
Definition: log.cpp:177
logger & warn()
Definition: log.cpp:312
void timestamps(bool t)
Definition: log.cpp:303
std::string sanitize_log(const std::string &logstr)
Definition: log.cpp:405
static severity strict_level_
Definition: log.cpp:302
const unsigned max_logs
Definition: log.hpp:75
void check_log_dir_writable()
Checks that a dummy file can be written to and deleted from the logs directory.
Definition: log.cpp:145
std::stringstream & log_to_chat()
Use this to show WML errors in the ingame chat.
Definition: log.cpp:517
const std::string log_file_prefix
Definition: log.hpp:67
std::map< std::string, severity > domain_map
Definition: log.cpp:300
static domain_map * domains
Definition: log.cpp:301
std::pair< const std::string, severity > logd
Definition: log.hpp:120
static bool strict_threw_
Definition: log.cpp:395
bool is_not_log_file(const std::string &fn)
Use the defined prefix and suffix to determine if a filename is a log file.
Definition: log.cpp:95
void precise_timestamps(bool pt)
Definition: log.cpp:304
logger & info()
Definition: log.cpp:318
bool set_log_domain_severity(const std::string &name, severity severity)
Definition: log.cpp:346
void set_strict_severity(severity severity)
Definition: log.cpp:387
std::string get_log_file_path()
Definition: log.cpp:284
const std::string log_file_suffix
Definition: log.hpp:68
std::ostringstream & operator<<(std::ostringstream &oss, const lg::severity severity)
Definition: log.cpp:89
constexpr auto filter
Definition: ranges.hpp:38
bool wildcard_string_match(const std::string &str, const std::string &match)
Match using '*' as any number of characters (including none), '+' as one or more characters,...
void erase_if(Container &container, const Predicate &predicate)
Convenience wrapper for using std::remove_if on a container.
Definition: general.hpp:106
std::string::const_iterator iterator
Definition: tokenizer.hpp:25
An exception object used when an IO error occurs.
Definition: filesystem.hpp:67
mock_char c