/tmp/bitcoin/src/logging.cpp
Line | Count | Source |
1 | | // Copyright (c) 2009-2010 Satoshi Nakamoto |
2 | | // Copyright (c) 2009-present The Bitcoin Core developers |
3 | | // Distributed under the MIT software license, see the accompanying |
4 | | // file COPYING or http://www.opensource.org/licenses/mit-license.php. |
5 | | |
6 | | #include <logging.h> |
7 | | #include <memusage.h> |
8 | | #include <util/check.h> |
9 | | #include <util/fs.h> |
10 | | #include <util/string.h> |
11 | | #include <util/threadnames.h> |
12 | | #include <util/time.h> |
13 | | |
14 | | #include <array> |
15 | | #include <cstring> |
16 | | #include <map> |
17 | | #include <optional> |
18 | | #include <utility> |
19 | | |
20 | | using util::Join; |
21 | | using util::RemovePrefixView; |
22 | | |
23 | | const char * const DEFAULT_DEBUGLOGFILE = "debug.log"; |
24 | | constexpr auto MAX_USER_SETABLE_SEVERITY_LEVEL{BCLog::Level::Info}; |
25 | | |
26 | | BCLog::Logger& LogInstance() |
27 | 11.6M | { |
28 | | /** |
29 | | * NOTE: the logger instances is leaked on exit. This is ugly, but will be |
30 | | * cleaned up by the OS/libc. Defining a logger as a global object doesn't work |
31 | | * since the order of destruction of static/global objects is undefined. |
32 | | * Consider if the logger gets destroyed, and then some later destructor calls |
33 | | * LogInfo, maybe indirectly, and you get a core dump at shutdown trying to |
34 | | * access the logger. When the shutdown sequence is fully audited and tested, |
35 | | * explicit destruction of these objects can be implemented by changing this |
36 | | * from a raw pointer to a std::unique_ptr. |
37 | | * Since the ~Logger() destructor is never called, the Logger class and all |
38 | | * its subclasses must have implicitly-defined destructors. |
39 | | * |
40 | | * This method of initialization was originally introduced in |
41 | | * ee3374234c60aba2cc4c5cd5cac1c0aefc2d817c. |
42 | | */ |
43 | 11.6M | static BCLog::Logger* g_logger{new BCLog::Logger()}; |
44 | 11.6M | return *g_logger; |
45 | 11.6M | } |
46 | | |
47 | | bool fLogIPs = DEFAULT_LOGIPS; |
48 | | |
49 | | static int FileWriteStr(std::string_view str, FILE *fp) |
50 | 5.23M | { |
51 | 5.23M | return fwrite(str.data(), 1, str.size(), fp); |
52 | 5.23M | } |
53 | | |
54 | | bool BCLog::Logger::StartLogging() |
55 | 1.78k | { |
56 | 1.78k | STDLOCK(m_cs); |
57 | | |
58 | 1.78k | assert(m_buffering); |
59 | 1.78k | assert(m_fileout == nullptr); |
60 | | |
61 | 1.78k | if (m_print_to_file) { |
62 | 1.78k | assert(!m_file_path.empty()); |
63 | 1.78k | m_fileout = fsbridge::fopen(m_file_path, "a"); |
64 | 1.78k | if (!m_fileout) { |
65 | 2 | return false; |
66 | 2 | } |
67 | | |
68 | 1.77k | setbuf(m_fileout, nullptr); // unbuffered |
69 | | |
70 | | // Add newlines to the logfile to distinguish this execution from the |
71 | | // last one. |
72 | 1.77k | FileWriteStr("\n\n\n\n\n", m_fileout); |
73 | 1.77k | } |
74 | | |
75 | | // dump buffered messages from before we opened the log |
76 | 1.78k | m_buffering = false; |
77 | 1.78k | if (m_buffer_lines_discarded > 0) { |
78 | 0 | LogPrint_({ |
79 | 0 | .category = BCLog::ALL, |
80 | 0 | .level = Level::Info, |
81 | 0 | .should_ratelimit = false, |
82 | 0 | .source_loc = SourceLocation{__func__}, |
83 | 0 | .message = strprintf("Early logging buffer overflowed, %d log lines discarded.", m_buffer_lines_discarded), |
84 | 0 | }); |
85 | 0 | } |
86 | 8.30k | while (!m_msgs_before_open.empty()) { |
87 | 6.52k | const auto& buflog = m_msgs_before_open.front(); |
88 | 6.52k | std::string s{Format(buflog)}; |
89 | 6.52k | m_msgs_before_open.pop_front(); |
90 | | |
91 | 6.52k | if (m_print_to_file) FileWriteStr(s, m_fileout); |
92 | 6.52k | if (m_print_to_console) fwrite(s.data(), 1, s.size(), stdout); |
93 | 6.52k | for (const auto& cb : m_print_callbacks) { |
94 | 0 | cb(s); |
95 | 0 | } |
96 | 6.52k | } |
97 | 1.78k | m_cur_buffer_memusage = 0; |
98 | 1.78k | if (m_print_to_console) fflush(stdout); |
99 | | |
100 | 1.78k | return true; |
101 | 1.78k | } |
102 | | |
103 | | void BCLog::Logger::DisconnectTestLogger() |
104 | 668 | { |
105 | 668 | STDLOCK(m_cs); |
106 | 668 | m_buffering = true; |
107 | 668 | if (m_fileout != nullptr) fclose(m_fileout); |
108 | 668 | m_fileout = nullptr; |
109 | 668 | m_print_callbacks.clear(); |
110 | 668 | m_max_buffer_memusage = DEFAULT_MAX_LOG_BUFFER; |
111 | 668 | m_cur_buffer_memusage = 0; |
112 | 668 | m_buffer_lines_discarded = 0; |
113 | 668 | m_msgs_before_open.clear(); |
114 | 668 | } |
115 | | |
116 | | void BCLog::Logger::DisableLogging() |
117 | 0 | { |
118 | 0 | { |
119 | 0 | STDLOCK(m_cs); |
120 | 0 | assert(m_buffering); |
121 | 0 | assert(m_print_callbacks.empty()); |
122 | 0 | } |
123 | 0 | m_print_to_file = false; |
124 | 0 | m_print_to_console = false; |
125 | 0 | StartLogging(); |
126 | 0 | } |
127 | | |
128 | | void BCLog::Logger::EnableCategory(BCLog::LogFlags flag) |
129 | 1.81k | { |
130 | 1.81k | m_categories |= flag; |
131 | 1.81k | } |
132 | | |
133 | | bool BCLog::Logger::EnableCategory(std::string_view str) |
134 | 1.80k | { |
135 | 1.80k | if (const auto flag{GetLogCategory(str)}) { |
136 | 1.80k | EnableCategory(*flag); |
137 | 1.80k | return true; |
138 | 1.80k | } |
139 | 1 | return false; |
140 | 1.80k | } |
141 | | |
142 | | void BCLog::Logger::DisableCategory(BCLog::LogFlags flag) |
143 | 4.74k | { |
144 | 4.74k | m_categories &= ~flag; |
145 | 4.74k | } |
146 | | |
147 | | bool BCLog::Logger::DisableCategory(std::string_view str) |
148 | 4.72k | { |
149 | 4.72k | if (const auto flag{GetLogCategory(str)}) { |
150 | 4.72k | DisableCategory(*flag); |
151 | 4.72k | return true; |
152 | 4.72k | } |
153 | 1 | return false; |
154 | 4.72k | } |
155 | | |
156 | | bool BCLog::Logger::WillLogCategory(BCLog::LogFlags category) const |
157 | 6.49M | { |
158 | 6.49M | return (m_categories.load(std::memory_order_relaxed) & category) != 0; |
159 | 6.49M | } |
160 | | |
161 | | bool BCLog::Logger::WillLogCategoryLevel(BCLog::LogFlags category, BCLog::Level level) const |
162 | 6.30M | { |
163 | | // Log messages at Info, Warning and Error level unconditionally, so that |
164 | | // important troubleshooting information doesn't get lost. |
165 | 6.30M | if (level >= BCLog::Level::Info) return true; |
166 | | |
167 | 6.30M | if (!WillLogCategory(category)) return false; |
168 | | |
169 | 5.24M | STDLOCK(m_cs); |
170 | 5.24M | const auto it{m_category_log_levels.find(category)}; |
171 | 18.4E | return level >= (it == m_category_log_levels.end() ? LogLevel() : it->second); |
172 | 6.30M | } |
173 | | |
174 | | bool BCLog::Logger::DefaultShrinkDebugFile() const |
175 | 1.11k | { |
176 | 1.11k | return m_categories == BCLog::NONE; |
177 | 1.11k | } |
178 | | |
179 | | static const std::map<std::string, BCLog::LogFlags, std::less<>> LOG_CATEGORIES_BY_STR{ |
180 | | {"net", BCLog::NET}, |
181 | | {"tor", BCLog::TOR}, |
182 | | {"mempool", BCLog::MEMPOOL}, |
183 | | {"http", BCLog::HTTP}, |
184 | | {"bench", BCLog::BENCH}, |
185 | | {"zmq", BCLog::ZMQ}, |
186 | | {"walletdb", BCLog::WALLETDB}, |
187 | | {"rpc", BCLog::RPC}, |
188 | | {"estimatefee", BCLog::ESTIMATEFEE}, |
189 | | {"addrman", BCLog::ADDRMAN}, |
190 | | {"selectcoins", BCLog::SELECTCOINS}, |
191 | | {"reindex", BCLog::REINDEX}, |
192 | | {"cmpctblock", BCLog::CMPCTBLOCK}, |
193 | | {"rand", BCLog::RAND}, |
194 | | {"prune", BCLog::PRUNE}, |
195 | | {"proxy", BCLog::PROXY}, |
196 | | {"mempoolrej", BCLog::MEMPOOLREJ}, |
197 | | {"libevent", BCLog::LIBEVENT}, |
198 | | {"coindb", BCLog::COINDB}, |
199 | | {"qt", BCLog::QT}, |
200 | | {"leveldb", BCLog::LEVELDB}, |
201 | | {"validation", BCLog::VALIDATION}, |
202 | | {"i2p", BCLog::I2P}, |
203 | | {"ipc", BCLog::IPC}, |
204 | | #ifdef DEBUG_LOCKCONTENTION |
205 | | {"lock", BCLog::LOCK}, |
206 | | #endif |
207 | | {"blockstorage", BCLog::BLOCKSTORAGE}, |
208 | | {"txreconciliation", BCLog::TXRECONCILIATION}, |
209 | | {"scan", BCLog::SCAN}, |
210 | | {"txpackages", BCLog::TXPACKAGES}, |
211 | | {"kernel", BCLog::KERNEL}, |
212 | | {"privatebroadcast", BCLog::PRIVBROADCAST}, |
213 | | }; |
214 | | |
215 | | static const std::unordered_map<BCLog::LogFlags, std::string> LOG_CATEGORIES_BY_FLAG{ |
216 | | // Swap keys and values from LOG_CATEGORIES_BY_STR. |
217 | 2.58k | [](const auto& in) { |
218 | 2.58k | std::unordered_map<BCLog::LogFlags, std::string> out; |
219 | 80.0k | for (const auto& [k, v] : in) { |
220 | 80.0k | const bool inserted{out.emplace(v, k).second}; |
221 | 80.0k | assert(inserted); |
222 | 80.0k | } |
223 | 2.58k | return out; |
224 | 2.58k | }(LOG_CATEGORIES_BY_STR) |
225 | | }; |
226 | | |
227 | | std::optional<BCLog::LogFlags> GetLogCategory(std::string_view str) |
228 | 6.57k | { |
229 | 6.57k | if (str.empty() || str == "1" || str == "all") { |
230 | 1.79k | return BCLog::ALL; |
231 | 1.79k | } |
232 | 4.77k | auto it = LOG_CATEGORIES_BY_STR.find(str); |
233 | 4.77k | if (it != LOG_CATEGORIES_BY_STR.end()) { |
234 | 4.77k | return it->second; |
235 | 4.77k | } |
236 | 2 | return std::nullopt; |
237 | 4.77k | } |
238 | | |
239 | | std::string BCLog::Logger::LogLevelToStr(BCLog::Level level) |
240 | 456k | { |
241 | 456k | switch (level) { |
242 | 445k | case BCLog::Level::Trace: |
243 | 445k | return "trace"; |
244 | 3.69k | case BCLog::Level::Debug: |
245 | 3.69k | return "debug"; |
246 | 1.84k | case BCLog::Level::Info: |
247 | 1.84k | return "info"; |
248 | 1.76k | case BCLog::Level::Warning: |
249 | 1.76k | return "warning"; |
250 | 3.43k | case BCLog::Level::Error: |
251 | 3.43k | return "error"; |
252 | 456k | } |
253 | 456k | assert(false); |
254 | 0 | } |
255 | | |
256 | | static std::string LogCategoryToStr(BCLog::LogFlags category) |
257 | 4.84M | { |
258 | 4.84M | if (category == BCLog::ALL) { |
259 | 0 | return "all"; |
260 | 0 | } |
261 | 4.84M | auto it = LOG_CATEGORIES_BY_FLAG.find(category); |
262 | 4.84M | assert(it != LOG_CATEGORIES_BY_FLAG.end()); |
263 | 4.84M | return it->second; |
264 | 4.84M | } |
265 | | |
266 | | static std::optional<BCLog::Level> GetLogLevel(std::string_view level_str) |
267 | 1.80k | { |
268 | 1.80k | if (level_str == "trace") { |
269 | 1.79k | return BCLog::Level::Trace; |
270 | 1.79k | } else if (level_str == "debug") { |
271 | 2 | return BCLog::Level::Debug; |
272 | 4 | } else if (level_str == "info") { |
273 | 2 | return BCLog::Level::Info; |
274 | 2 | } else if (level_str == "warning") { |
275 | 0 | return BCLog::Level::Warning; |
276 | 2 | } else if (level_str == "error") { |
277 | 0 | return BCLog::Level::Error; |
278 | 2 | } else { |
279 | 2 | return std::nullopt; |
280 | 2 | } |
281 | 1.80k | } |
282 | | |
283 | | std::vector<LogCategory> BCLog::Logger::LogCategoriesList() const |
284 | 6.02k | { |
285 | 6.02k | std::vector<LogCategory> ret; |
286 | 6.02k | ret.reserve(LOG_CATEGORIES_BY_STR.size()); |
287 | 186k | for (const auto& [category, flag] : LOG_CATEGORIES_BY_STR) { |
288 | 186k | ret.push_back(LogCategory{.category = category, .active = WillLogCategory(flag)}); |
289 | 186k | } |
290 | 6.02k | return ret; |
291 | 6.02k | } |
292 | | |
293 | | /** Log severity levels that can be selected by the user. */ |
294 | | static constexpr std::array<BCLog::Level, 3> LogLevelsList() |
295 | 1.84k | { |
296 | 1.84k | return {BCLog::Level::Info, BCLog::Level::Debug, BCLog::Level::Trace}; |
297 | 1.84k | } |
298 | | |
299 | | std::string BCLog::Logger::LogLevelsString() const |
300 | 1.84k | { |
301 | 1.84k | const auto& levels = LogLevelsList(); |
302 | 5.54k | return Join(std::vector<BCLog::Level>{levels.begin(), levels.end()}, ", ", [](BCLog::Level level) { return LogLevelToStr(level); }); |
303 | 1.84k | } |
304 | | |
305 | | std::string BCLog::Logger::LogTimestampStr(SystemClock::time_point now, std::chrono::seconds mocktime) const |
306 | 5.23M | { |
307 | 5.23M | std::string strStamped; |
308 | | |
309 | 5.23M | if (!m_log_timestamps) |
310 | 91 | return strStamped; |
311 | | |
312 | 5.23M | const auto now_seconds{std::chrono::time_point_cast<std::chrono::seconds>(now)}; |
313 | 5.23M | strStamped = FormatISO8601DateTime(TicksSinceEpoch<std::chrono::seconds>(now_seconds)); |
314 | 5.23M | if (m_log_time_micros && !strStamped.empty()) { |
315 | 5.23M | strStamped.pop_back(); |
316 | 5.23M | strStamped += strprintf(".%06dZ", Ticks<std::chrono::microseconds>(now - now_seconds)); |
317 | 5.23M | } |
318 | 5.23M | if (mocktime > 0s) { |
319 | 1.12M | strStamped += " (mocktime: " + FormatISO8601DateTime(count_seconds(mocktime)) + ")"; |
320 | 1.12M | } |
321 | 5.23M | strStamped += ' '; |
322 | | |
323 | 5.23M | return strStamped; |
324 | 5.23M | } |
325 | | |
326 | | namespace BCLog { |
327 | | /** Belts and suspenders: make sure outgoing log messages don't contain |
328 | | * potentially suspicious characters, such as terminal control codes. |
329 | | * |
330 | | * This escapes control characters except newline ('\n') in C syntax. |
331 | | * It escapes instead of removes them to still allow for troubleshooting |
332 | | * issues where they accidentally end up in strings. |
333 | | */ |
334 | 5.23M | std::string LogEscapeMessage(std::string_view str) { |
335 | 5.23M | std::string ret; |
336 | 455M | for (char ch_in : str) { |
337 | 455M | uint8_t ch = (uint8_t)ch_in; |
338 | 455M | if ((ch >= 32 || ch == '\n') && ch != '\x7f') { |
339 | 455M | ret += ch_in; |
340 | 455M | } else { |
341 | 6.82k | ret += strprintf("\\x%02x", ch); |
342 | 6.82k | } |
343 | 455M | } |
344 | 5.23M | return ret; |
345 | 5.23M | } |
346 | | } // namespace BCLog |
347 | | |
348 | | std::string BCLog::Logger::GetLogPrefix(BCLog::LogFlags category, BCLog::Level level) const |
349 | 5.23M | { |
350 | 5.23M | if (category == LogFlags::NONE) category = LogFlags::ALL; |
351 | | |
352 | 5.23M | const bool has_category{m_always_print_category_level || category != LogFlags::ALL}; |
353 | | |
354 | | // If there is no category, Info is implied |
355 | 5.23M | if (!has_category && level == Level::Info) return {}; |
356 | | |
357 | 4.84M | std::string s{"["}; |
358 | 4.84M | if (has_category) { |
359 | 4.84M | s += LogCategoryToStr(category); |
360 | 4.84M | } |
361 | | |
362 | 4.84M | if (m_always_print_category_level || !has_category || level != Level::Debug) { |
363 | | // If there is a category, Debug is implied, so don't add the level |
364 | | |
365 | | // Only add separator if we have a category |
366 | 448k | if (has_category) s += ":"; |
367 | 448k | s += Logger::LogLevelToStr(level); |
368 | 448k | } |
369 | | |
370 | 4.84M | s += "] "; |
371 | 4.84M | return s; |
372 | 5.23M | } |
373 | | |
374 | | static size_t MemUsage(const util::log::Entry& log) |
375 | 29.1k | { |
376 | 29.1k | return memusage::DynamicUsage(log.message) + |
377 | 29.1k | memusage::DynamicUsage(log.thread_name) + |
378 | 29.1k | memusage::MallocUsage(sizeof(memusage::list_node<util::log::Entry>)); |
379 | 29.1k | } |
380 | | |
381 | | BCLog::LogRateLimiter::LogRateLimiter(uint64_t max_bytes, std::chrono::seconds reset_window) |
382 | 3 | : m_max_bytes{max_bytes}, m_reset_window{reset_window} {} |
383 | | |
384 | | std::shared_ptr<BCLog::LogRateLimiter> BCLog::LogRateLimiter::Create( |
385 | | SchedulerFunction&& scheduler_func, uint64_t max_bytes, std::chrono::seconds reset_window) |
386 | 3 | { |
387 | 3 | auto limiter{std::shared_ptr<LogRateLimiter>(new LogRateLimiter(max_bytes, reset_window))}; |
388 | 3 | std::weak_ptr<LogRateLimiter> weak_limiter{limiter}; |
389 | 3 | auto reset = [weak_limiter] { |
390 | 2 | if (auto shared_limiter{weak_limiter.lock()}) shared_limiter->Reset(); |
391 | 2 | }; |
392 | 3 | scheduler_func(reset, limiter->m_reset_window); |
393 | 3 | return limiter; |
394 | 3 | } |
395 | | |
396 | | BCLog::LogRateLimiter::Status BCLog::LogRateLimiter::Consume( |
397 | | const SourceLocation& source_loc, |
398 | | const std::string& str) |
399 | 89 | { |
400 | 89 | STDLOCK(m_mutex); |
401 | 89 | auto& stats{m_source_locations.try_emplace(source_loc, m_max_bytes).first->second}; |
402 | 89 | Status status{stats.m_dropped_bytes > 0 ? Status::STILL_SUPPRESSED : Status::UNSUPPRESSED}; |
403 | | |
404 | 89 | if (!stats.Consume(str.size()) && status == Status::UNSUPPRESSED) { |
405 | 3 | status = Status::NEWLY_SUPPRESSED; |
406 | 3 | m_suppression_active = true; |
407 | 3 | } |
408 | | |
409 | 89 | return status; |
410 | 89 | } |
411 | | |
412 | | std::string BCLog::Logger::Format(const util::log::Entry& entry) const |
413 | 5.23M | { |
414 | 5.23M | std::string result{LogTimestampStr(entry.timestamp, entry.mocktime)}; |
415 | | |
416 | 5.23M | if (m_log_threadnames) { |
417 | 5.23M | result += strprintf("[%s] ", (entry.thread_name.empty() ? "unknown" : entry.thread_name)); |
418 | 5.23M | } |
419 | | |
420 | 5.23M | if (m_log_sourcelocations) { |
421 | 5.23M | result += strprintf("[%s:%d] [%s] ", RemovePrefixView(entry.source_loc.file_name(), "./"), entry.source_loc.line(), entry.source_loc.function_name_short()); |
422 | 5.23M | } |
423 | | |
424 | 5.23M | result += GetLogPrefix(static_cast<LogFlags>(entry.category), entry.level); |
425 | 5.23M | result += LogEscapeMessage(entry.message); |
426 | | |
427 | 5.23M | if (!result.ends_with('\n')) result += '\n'; |
428 | 5.23M | return result; |
429 | 5.23M | } |
430 | | |
431 | | void BCLog::Logger::LogPrint(util::log::Entry entry) |
432 | 5.25M | { |
433 | 5.25M | STDLOCK(m_cs); |
434 | 5.25M | return LogPrint_(std::move(entry)); |
435 | 5.25M | } |
436 | | |
437 | | // NOLINTNEXTLINE(misc-no-recursion) |
438 | | void BCLog::Logger::LogPrint_(util::log::Entry entry) |
439 | 5.25M | { |
440 | 5.25M | if (m_buffering) { |
441 | 29.1k | { |
442 | 29.1k | m_cur_buffer_memusage += MemUsage(entry); |
443 | 29.1k | m_msgs_before_open.push_back(std::move(entry)); |
444 | 29.1k | } |
445 | | |
446 | 29.1k | while (m_cur_buffer_memusage > m_max_buffer_memusage) { |
447 | 0 | if (m_msgs_before_open.empty()) { |
448 | 0 | m_cur_buffer_memusage = 0; |
449 | 0 | break; |
450 | 0 | } |
451 | 0 | m_cur_buffer_memusage -= MemUsage(m_msgs_before_open.front()); |
452 | 0 | m_msgs_before_open.pop_front(); |
453 | 0 | ++m_buffer_lines_discarded; |
454 | 0 | } |
455 | | |
456 | 29.1k | return; |
457 | 29.1k | } |
458 | | |
459 | 5.22M | std::string str_prefixed{Format(entry)}; |
460 | 5.22M | bool ratelimit{false}; |
461 | 5.22M | if (entry.should_ratelimit && m_limiter) { |
462 | 81 | auto status{m_limiter->Consume(entry.source_loc, str_prefixed)}; |
463 | 81 | if (status == LogRateLimiter::Status::NEWLY_SUPPRESSED) { |
464 | | // NOLINTNEXTLINE(misc-no-recursion) |
465 | 1 | LogPrint_({ |
466 | 1 | .category = LogFlags::ALL, |
467 | 1 | .level = Level::Warning, |
468 | 1 | .should_ratelimit = false, // with should_ratelimit=false, this cannot lead to infinite recursion |
469 | 1 | .source_loc = SourceLocation{__func__}, |
470 | 1 | .message = strprintf( |
471 | 1 | "Excessive logging detected from %s:%d (%s): >%d bytes logged during " |
472 | 1 | "the last time window of %is. Suppressing logging to disk from this " |
473 | 1 | "source location until time window resets. Console logging " |
474 | 1 | "unaffected. Last log entry.", |
475 | 1 | entry.source_loc.file_name(), entry.source_loc.line(), entry.source_loc.function_name_short(), |
476 | 1 | m_limiter->m_max_bytes, |
477 | 1 | Ticks<std::chrono::seconds>(m_limiter->m_reset_window)), |
478 | 1 | }); |
479 | 80 | } else if (status == LogRateLimiter::Status::STILL_SUPPRESSED) { |
480 | 1 | ratelimit = true; |
481 | 1 | } |
482 | 81 | } |
483 | | |
484 | | // To avoid confusion caused by dropped log messages when debugging an issue, |
485 | | // we prefix log lines with "[*]" when there are any suppressed source locations. |
486 | 5.22M | if (m_limiter && m_limiter->SuppressionsActive()) { |
487 | 4 | str_prefixed.insert(0, "[*] "); |
488 | 4 | } |
489 | | |
490 | 5.22M | if (m_print_to_console) { |
491 | | // print to console |
492 | 419k | fwrite(str_prefixed.data(), 1, str_prefixed.size(), stdout); |
493 | 419k | fflush(stdout); |
494 | 419k | } |
495 | 5.22M | for (const auto& cb : m_print_callbacks) { |
496 | 1.05k | cb(str_prefixed); |
497 | 1.05k | } |
498 | 5.22M | if (m_print_to_file && !ratelimit) { |
499 | 5.22M | assert(m_fileout != nullptr); |
500 | | |
501 | | // reopen the log file, if requested |
502 | 5.22M | if (m_reopen_file) { |
503 | 6 | m_reopen_file = false; |
504 | 6 | FILE* new_fileout = fsbridge::fopen(m_file_path, "a"); |
505 | 6 | if (new_fileout) { |
506 | 6 | setbuf(new_fileout, nullptr); // unbuffered |
507 | 6 | fclose(m_fileout); |
508 | 6 | m_fileout = new_fileout; |
509 | 6 | } |
510 | 6 | } |
511 | 5.22M | FileWriteStr(str_prefixed, m_fileout); |
512 | 5.22M | } |
513 | 5.22M | } |
514 | | |
515 | | void BCLog::Logger::ShrinkDebugFile() |
516 | 0 | { |
517 | | // Amount of debug.log to save at end when shrinking (must fit in memory) |
518 | 0 | constexpr size_t RECENT_DEBUG_HISTORY_SIZE = 10 * 1000000; |
519 | |
|
520 | 0 | assert(!m_file_path.empty()); |
521 | | |
522 | | // Scroll debug.log if it's getting too big |
523 | 0 | FILE* file = fsbridge::fopen(m_file_path, "r"); |
524 | | |
525 | | // Special files (e.g. device nodes) may not have a size. |
526 | 0 | size_t log_size = 0; |
527 | 0 | try { |
528 | 0 | log_size = fs::file_size(m_file_path); |
529 | 0 | } catch (const fs::filesystem_error&) {} |
530 | | |
531 | | // If debug.log file is more than 10% bigger the RECENT_DEBUG_HISTORY_SIZE |
532 | | // trim it down by saving only the last RECENT_DEBUG_HISTORY_SIZE bytes |
533 | 0 | if (file && log_size > 11 * (RECENT_DEBUG_HISTORY_SIZE / 10)) |
534 | 0 | { |
535 | | // Restart the file with some of the end |
536 | 0 | std::vector<char> vch(RECENT_DEBUG_HISTORY_SIZE, 0); |
537 | 0 | if (fseek(file, -((long)vch.size()), SEEK_END)) { |
538 | 0 | LogWarning("Failed to shrink debug log file: fseek(...) failed"); |
539 | 0 | fclose(file); |
540 | 0 | return; |
541 | 0 | } |
542 | 0 | int nBytes = fread(vch.data(), 1, vch.size(), file); |
543 | 0 | fclose(file); |
544 | |
|
545 | 0 | file = fsbridge::fopen(m_file_path, "w"); |
546 | 0 | if (file) |
547 | 0 | { |
548 | 0 | fwrite(vch.data(), 1, nBytes, file); |
549 | 0 | fclose(file); |
550 | 0 | } |
551 | 0 | } |
552 | 0 | else if (file != nullptr) |
553 | 0 | fclose(file); |
554 | 0 | } |
555 | | |
556 | | void BCLog::LogRateLimiter::Reset() |
557 | 3 | { |
558 | 3 | decltype(m_source_locations) source_locations; |
559 | 3 | { |
560 | 3 | STDLOCK(m_mutex); |
561 | 3 | source_locations.swap(m_source_locations); |
562 | 3 | m_suppression_active = false; |
563 | 3 | } |
564 | 4 | for (const auto& [source_loc, stats] : source_locations) { |
565 | 4 | if (stats.m_dropped_bytes == 0) continue; |
566 | 3 | LogPrintLevel_( |
567 | 3 | LogFlags::ALL, Level::Warning, /*should_ratelimit=*/false, |
568 | 3 | "Restarting logging from %s:%d (%s): %d bytes were dropped during the last %ss.", |
569 | 3 | source_loc.file_name(), source_loc.line(), source_loc.function_name_short(), |
570 | 3 | stats.m_dropped_bytes, Ticks<std::chrono::seconds>(m_reset_window)); |
571 | 3 | } |
572 | 3 | } |
573 | | |
574 | | bool BCLog::LogRateLimiter::Stats::Consume(uint64_t bytes) |
575 | 92 | { |
576 | 92 | if (bytes > m_available_bytes) { |
577 | 6 | m_dropped_bytes += bytes; |
578 | 6 | m_available_bytes = 0; |
579 | 6 | return false; |
580 | 6 | } |
581 | | |
582 | 86 | m_available_bytes -= bytes; |
583 | 86 | return true; |
584 | 92 | } |
585 | | |
586 | | bool BCLog::Logger::SetLogLevel(std::string_view level_str) |
587 | 1.79k | { |
588 | 1.79k | const auto level = GetLogLevel(level_str); |
589 | 1.79k | if (!level.has_value() || level.value() > MAX_USER_SETABLE_SEVERITY_LEVEL) return false; |
590 | 1.79k | m_log_level = level.value(); |
591 | 1.79k | return true; |
592 | 1.79k | } |
593 | | |
594 | | bool BCLog::Logger::SetCategoryLogLevel(std::string_view category_str, std::string_view level_str) |
595 | 5 | { |
596 | 5 | const auto flag{GetLogCategory(category_str)}; |
597 | 5 | if (!flag) return false; |
598 | | |
599 | 5 | const auto level = GetLogLevel(level_str); |
600 | 5 | if (!level.has_value() || level.value() > MAX_USER_SETABLE_SEVERITY_LEVEL) return false; |
601 | | |
602 | 4 | STDLOCK(m_cs); |
603 | 4 | m_category_log_levels[*flag] = level.value(); |
604 | 4 | return true; |
605 | 5 | } |
606 | | |
607 | | bool util::log::ShouldLog(Category category, Level level) |
608 | 6.24M | { |
609 | 6.24M | return LogInstance().WillLogCategoryLevel(static_cast<BCLog::LogFlags>(category), level); |
610 | 6.24M | } |
611 | | |
612 | | void util::log::Log(util::log::Entry entry) |
613 | 5.25M | { |
614 | 5.25M | BCLog::Logger& logger{LogInstance()}; |
615 | 5.25M | if (logger.Enabled()) { |
616 | 5.25M | logger.LogPrint(std::move(entry)); |
617 | 5.25M | } |
618 | 5.25M | } |