Line | Branch | Exec | Source |
---|---|---|---|
1 | #pragma once | ||
2 | |||
3 | #include <fmt/color.h> | ||
4 | #include <fmt/format.h> | ||
5 | #include <fmt/ranges.h> | ||
6 | |||
7 | #include <functional> | ||
8 | #include <unordered_map> | ||
9 | |||
10 | namespace iguana { | ||
11 | |||
12 | /// @brief Simple logger service | ||
13 | /// | ||
14 | /// - Each algorithm instance should own a `Logger` instance | ||
15 | /// - The user may control the log level of each `Logger`, thus the log level of each algorithm | ||
16 | /// - Errors and warnings print to `stderr`, whereas all other levels print to `stdout` | ||
17 | class Logger | ||
18 | { | ||
19 | |||
20 | friend class Object; | ||
21 | |||
22 | public: | ||
23 | |||
24 | /// These are the available log levels, from lowest to highest: | ||
25 | /// - `trace`: the most verbose level, used for fine-grained printouts for each event | ||
26 | /// - `debug`: less verbose printout, expected to be less frequent than `trace` | ||
27 | /// - `info`: the least verbose printout; this is the default level | ||
28 | /// - `quiet`: use this level to only allow warnings and errors, silencing all other printouts | ||
29 | /// - `warn`: an issue that may or may not be critical | ||
30 | /// - `error`: an issue that is likely critical | ||
31 | /// - `silent`: use this level to silence **all** printouts (use it at your own risk!) | ||
32 | enum Level { | ||
33 | trace, | ||
34 | debug, | ||
35 | info, | ||
36 | quiet, | ||
37 | warn, | ||
38 | error, | ||
39 | silent | ||
40 | }; | ||
41 | |||
42 | /// The default log level | ||
43 | static Level const DEFAULT_LEVEL = info; | ||
44 | |||
45 | /// @param name the name of this logger instance, which will be include in all of its printouts | ||
46 | /// @param lev the log level | ||
47 | /// @param enable_style if true, certain printouts will be styled with color and emphasis | ||
48 | Logger(std::string_view name = "log", Level const lev = DEFAULT_LEVEL, bool const enable_style = true); | ||
49 | 204 | ~Logger() {} | |
50 | |||
51 | /// Set the log level to this level. Log messages with a lower level will not be printed. | ||
52 | /// @see `Logger::Level` for available levels. | ||
53 | /// @param lev the log level name | ||
54 | void SetLevel(std::string_view lev); | ||
55 | |||
56 | /// Set the log level to this level. Log messages with a lower level will not be printed. | ||
57 | /// @see `Logger::Level` for available levels. | ||
58 | /// @param lev the log level | ||
59 | void SetLevel(Level const lev); | ||
60 | |||
61 | /// Get the current log level | ||
62 | /// @returns the log level | ||
63 | Level GetLevel(); | ||
64 | |||
65 | /// Enable styled log printouts, with color and emphasis | ||
66 | void EnableStyle(); | ||
67 | |||
68 | /// Disable styled log printout color and emphasis | ||
69 | void DisableStyle(); | ||
70 | |||
71 | /// Generate a header for a printout | ||
72 | /// @param message the header message | ||
73 | /// @param width the width of the header in number of characters | ||
74 | /// @returns the header string | ||
75 | static std::string Header(std::string_view message, int const width = 50); | ||
76 | |||
77 | /// Printout a log message at the `trace` level @see `Logger::Print` for more details | ||
78 | template <typename... VALUES> | ||
79 |
2/4✓ Branch 0 (42→43) taken 16618 times.
✗ Branch 1 (42→48) not taken.
✓ Branch 2 (44→45) taken 1665 times.
✗ Branch 3 (44→48) not taken.
|
46026 | void Trace(std::string_view message, const VALUES... vals) const { Print(trace, message, vals...); } |
80 | /// Printout a log message at the `debug` level @see `Logger::Print` for more details | ||
81 | template <typename... VALUES> | ||
82 |
1/2✓ Branch 0 (3→4) taken 419 times.
✗ Branch 1 (3→6) not taken.
|
72634 | void Debug(std::string_view message, const VALUES... vals) const { Print(debug, message, vals...); } |
83 | /// Printout a log message at the `info` level @see `Logger::Print` for more details | ||
84 | template <typename... VALUES> | ||
85 |
4/20✓ Branch 0 (19→20) taken 2 times.
✗ Branch 1 (19→45) not taken.
✓ Branch 2 (25→26) taken 2 times.
✗ Branch 3 (25→45) not taken.
✗ Branch 4 (26→27) not taken.
✓ Branch 5 (26→38) taken 2 times.
✓ Branch 6 (41→42) taken 2 times.
✗ Branch 7 (41→43) not taken.
✗ Branch 8 (27→28) not taken.
✗ Branch 9 (27→67) not taken.
✗ Branch 10 (28→29) not taken.
✗ Branch 11 (28→67) not taken.
✗ Branch 12 (29→30) not taken.
✗ Branch 13 (29→67) not taken.
✗ Branch 14 (31→32) not taken.
✗ Branch 15 (31→67) not taken.
✗ Branch 16 (35→36) not taken.
✗ Branch 17 (35→67) not taken.
✗ Branch 18 (77→78) not taken.
✗ Branch 19 (77→96) not taken.
|
15 | void Info(std::string_view message, const VALUES... vals) const { Print(info, message, vals...); } |
86 | /// Printout a log message at the `warn` level @see `Logger::Print` for more details | ||
87 | template <typename... VALUES> | ||
88 |
1/4✓ Branch 0 (20→21) taken 24 times.
✗ Branch 1 (20→45) not taken.
✗ Branch 2 (33→34) not taken.
✗ Branch 3 (33→67) not taken.
|
32 | void Warn(std::string_view message, const VALUES... vals) const { Print(warn, message, vals...); } |
89 | /// Printout a log message at the `error` level @see `Logger::Print` for more details | ||
90 | template <typename... VALUES> | ||
91 |
2/28✓ Branch 0 (3→4) taken 13 times.
✗ Branch 1 (3→6) not taken.
✓ Branch 2 (51→52) taken 2 times.
✗ Branch 3 (51→55) not taken.
✗ Branch 4 (45→46) not taken.
✗ Branch 5 (45→51) not taken.
✗ Branch 6 (49→50) not taken.
✗ Branch 7 (49→53) not taken.
✗ Branch 8 (45→46) not taken.
✗ Branch 9 (45→51) not taken.
✗ Branch 10 (49→50) not taken.
✗ Branch 11 (49→53) not taken.
✗ Branch 12 (17→18) not taken.
✗ Branch 13 (17→23) not taken.
✗ Branch 14 (21→22) not taken.
✗ Branch 15 (21→25) not taken.
✗ Branch 16 (15→16) not taken.
✗ Branch 17 (15→21) not taken.
✗ Branch 18 (19→20) not taken.
✗ Branch 19 (19→23) not taken.
✗ Branch 20 (15→16) not taken.
✗ Branch 21 (15→21) not taken.
✗ Branch 22 (19→20) not taken.
✗ Branch 23 (19→23) not taken.
✗ Branch 24 (31→32) not taken.
✗ Branch 25 (31→37) not taken.
✗ Branch 26 (35→36) not taken.
✗ Branch 27 (35→39) not taken.
|
26 | void Error(std::string_view message, const VALUES... vals) const { Print(error, message, vals...); } |
92 | |||
93 | /// Printout a log message at the specified level. The message will only print if `lev` is at least as high as the current level of | ||
94 | /// this `Logger` instance, as set by `Logger::SetLevel`. | ||
95 | /// @param lev the log level for this message | ||
96 | /// @param message the message to print; this may be a format string, as in `fmt::format` | ||
97 | /// @param vals values for the format string `message` | ||
98 | template <typename... VALUES> | ||
99 | 101877 | void Print(Level const lev, std::string_view message, const VALUES... vals) const | |
100 | { | ||
101 |
2/2✓ Branch 0 (2→3) taken 951 times.
✓ Branch 1 (2→37) taken 100926 times.
|
101877 | if(lev >= m_level) { |
102 |
1/2✓ Branch 0 (3→4) taken 951 times.
✗ Branch 1 (3→35) not taken.
|
951 | if(auto it{m_level_names.find(lev)}; it != m_level_names.end()) { |
103 | std::function<std::string(std::string)> style = [](std::string s) | ||
104 | 18 | { return fmt::format("[{}]", s); }; | |
105 |
2/2✓ Branch 0 (4→5) taken 942 times.
✓ Branch 1 (4→12) taken 9 times.
|
951 | if(m_enable_style) { |
106 |
3/3✓ Branch 0 (5→6) taken 23 times.
✓ Branch 1 (5→8) taken 27 times.
✓ Branch 2 (5→10) taken 892 times.
|
942 | switch(lev) { |
107 | 23 | case warn: | |
108 | 69 | style = [](std::string s) | |
109 | 46 | { return fmt::format("[{}]", fmt::styled(s, fmt::emphasis::bold | fmt::fg(fmt::terminal_color::magenta))); }; | |
110 | 23 | break; | |
111 | 27 | case error: | |
112 | 81 | style = [](std::string s) | |
113 | 54 | { return fmt::format("[{}]", fmt::styled(s, fmt::emphasis::bold | fmt::fg(fmt::terminal_color::red))); }; | |
114 | 27 | break; | |
115 | 892 | default: | |
116 | 2676 | style = [](std::string s) | |
117 | 1784 | { return fmt::format("[{}]", fmt::styled(s, fmt::emphasis::bold)); }; | |
118 | } | ||
119 | } | ||
120 |
3/4✓ Branch 0 (23→24) taken 52 times.
✓ Branch 1 (23→25) taken 899 times.
✓ Branch 2 (26→27) taken 42 times.
✗ Branch 3 (26→45) not taken.
|
1858 | fmt::print( |
121 | lev >= warn ? stderr : stdout, | ||
122 |
3/6✓ Branch 0 (22→23) taken 951 times.
✗ Branch 1 (22→40) not taken.
✓ Branch 2 (32→33) taken 949 times.
✗ Branch 3 (32→34) not taken.
✓ Branch 4 (48→49) taken 2 times.
✗ Branch 5 (48→50) not taken.
|
3804 | fmt::runtime(fmt::format("{} {} {}\n", style(it->second), style(m_name), message)), |
123 | vals...); | ||
124 | } | ||
125 | else { | ||
126 | ✗ | Warn("Logger::Print called with unknown log level '{}'; printing as error instead", static_cast<int>(lev)); // FIXME: static_cast -> fmt::underlying, but needs new version of fmt | |
127 | ✗ | Error(message, vals...); | |
128 | } | ||
129 | } | ||
130 | 101875 | } | |
131 | |||
132 | private: | ||
133 | |||
134 | /// The name of this logger, which is included in all printouts | ||
135 | std::string m_name; | ||
136 | |||
137 | /// The current log level for this instance | ||
138 | Level m_level; | ||
139 | |||
140 | /// Association of the log level to its name | ||
141 | std::unordered_map<Level, std::string> m_level_names; | ||
142 | |||
143 | /// If true, style the printouts | ||
144 | bool m_enable_style; | ||
145 | }; | ||
146 | } | ||
147 |