log/macros.rs
1// Copyright 2014-2015 The Rust Project Developers. See the COPYRIGHT
2// file at the top-level directory of this distribution and at
3// http://rust-lang.org/COPYRIGHT.
4//
5// Licensed under the Apache License, Version 2.0 <LICENSE-APACHE or
6// http://www.apache.org/licenses/LICENSE-2.0> or the MIT license
7// <LICENSE-MIT or http://opensource.org/licenses/MIT>, at your
8// option. This file may not be copied, modified, or distributed
9// except according to those terms.
10
11/// The standard logging macro.
12///
13/// This macro will generically log with the specified `Level` and `format!`
14/// based argument list.
15///
16/// ```
17/// use log::{log, Level};
18///
19/// let data = (42, "Forty-two");
20/// let private_data = "private";
21///
22/// log!(Level::Error, "Received errors: {}, {}", data.0, data.1);
23/// ```
24///
25/// Optionally, you can specify a `target` argument to attach a specific target
26/// to the log record. By default, the target is the module path of the caller.
27///
28/// ```
29/// use log::{log, Level};
30///
31/// let data = (42, "Forty-two");
32/// let private_data = "private";
33///
34/// log!(
35/// target: "app_events",
36/// Level::Error,
37/// "Received errors: {}, {}",
38/// data.0, data.1
39/// );
40/// ```
41///
42/// And optionally, you can specify a `logger` argument to use a specific logger
43/// instead of the default global logger.
44///
45/// ```
46/// # struct MyLogger {}
47/// # impl Log for MyLogger {
48/// # fn enabled(&self, _metadata: &log::Metadata) -> bool {
49/// # false
50/// # }
51/// # fn log(&self, _record: &log::Record) {}
52/// # fn flush(&self) {}
53/// # }
54/// use log::{log, Level, Log};
55///
56/// let data = (42, "Forty-two");
57/// let private_data = "private";
58///
59/// let my_logger = MyLogger {};
60/// log!(
61/// logger: my_logger,
62/// Level::Error,
63/// "Received errors: {}, {}",
64/// data.0, data.1
65/// );
66/// ```
67///
68/// The `logger` argument accepts a value that implements the `Log` trait. The value
69/// will be borrowed within the macro.
70///
71/// Note that the global level set via Cargo features, or through `set_max_level` will
72/// still apply, even when a custom logger is supplied with the `logger` argument.
73#[macro_export]
74#[clippy::format_args]
75macro_rules! log {
76 // log!(logger: my_logger, target: "my_target", Level::Info, "a {} event", "log");
77 (logger: $logger:expr, target: $target:expr, $lvl:expr, $($arg:tt)+) => ({
78 $crate::__log!(
79 logger: $crate::__log_logger!($logger),
80 target: $target,
81 $lvl,
82 $($arg)+
83 )
84 });
85
86 // log!(logger: my_logger, Level::Info, "a log event")
87 (logger: $logger:expr, $lvl:expr, $($arg:tt)+) => ({
88 $crate::__log!(
89 logger: $crate::__log_logger!($logger),
90 target: $crate::__private_api::module_path!(),
91 $lvl,
92 $($arg)+
93 )
94 });
95
96 // log!(target: "my_target", Level::Info, "a log event")
97 (target: $target:expr, $lvl:expr, $($arg:tt)+) => ({
98 $crate::__log!(
99 logger: $crate::__log_logger!(__log_global_logger),
100 target: $target,
101 $lvl,
102 $($arg)+
103 )
104 });
105
106 // log!(Level::Info, "a log event")
107 ($lvl:expr, $($arg:tt)+) => ({
108 $crate::__log!(
109 logger: $crate::__log_logger!(__log_global_logger),
110 target: $crate::__private_api::module_path!(),
111 $lvl,
112 $($arg)+
113 )
114 });
115}
116
117#[doc(hidden)]
118#[macro_export]
119macro_rules! __log {
120 // log!(logger: my_logger, target: "my_target", Level::Info, key1:? = 42, key2 = true; "a {} event", "log");
121 (logger: $logger:expr, target: $target:expr, $lvl:expr, $($key:tt $(:$capture:tt)? $(= $value:expr)?),+; $($arg:tt)+) => ({
122 let lvl = $lvl;
123 if lvl <= $crate::STATIC_MAX_LEVEL && lvl <= $crate::max_level() {
124 $crate::__private_api::log(
125 $logger,
126 $crate::__private_api::format_args!($($arg)+),
127 lvl,
128 &($target, $crate::__private_api::module_path!(), $crate::__private_api::loc()),
129 &[$(($crate::__log_key!($key), $crate::__log_value!($key $(:$capture)* = $($value)*))),+] as &[_],
130 );
131 }
132 });
133
134 // log!(logger: my_logger, target: "my_target", Level::Info, "a {} event", "log");
135 (logger: $logger:expr, target: $target:expr, $lvl:expr, $($arg:tt)+) => ({
136 let lvl = $lvl;
137 if lvl <= $crate::STATIC_MAX_LEVEL && lvl <= $crate::max_level() {
138 $crate::__private_api::log(
139 $logger,
140 $crate::__private_api::format_args!($($arg)+),
141 lvl,
142 &($target, $crate::__private_api::module_path!(), $crate::__private_api::loc()),
143 (),
144 );
145 }
146 });
147}
148
149/// Logs a message at the error level.
150///
151/// # Examples
152///
153/// ```
154/// use log::error;
155///
156/// # let my_logger = log::__private_api::GlobalLogger;
157/// let (err_info, port) = ("No connection", 22);
158///
159/// error!("Error: {err_info} on port {port}");
160/// error!(target: "app_events", "App Error: {err_info}, Port: {port}");
161/// error!(logger: my_logger, "App Error: {err_info}, Port: {port}");
162/// ```
163#[macro_export]
164#[clippy::format_args]
165macro_rules! error {
166 // error!(logger: my_logger, target: "my_target", key1 = 42, key2 = true; "a {} event", "log")
167 // error!(logger: my_logger, target: "my_target", "a {} event", "log")
168 (logger: $logger:expr, target: $target:expr, $($arg:tt)+) => ({
169 $crate::log!(logger: $crate::__log_logger!($logger), target: $target, $crate::Level::Error, $($arg)+)
170 });
171
172 // error!(logger: my_logger, key1 = 42, key2 = true; "a {} event", "log")
173 // error!(logger: my_logger, "a {} event", "log")
174 (logger: $logger:expr, $($arg:tt)+) => ({
175 $crate::log!(logger: $crate::__log_logger!($logger), $crate::Level::Error, $($arg)+)
176 });
177
178 // error!(target: "my_target", key1 = 42, key2 = true; "a {} event", "log")
179 // error!(target: "my_target", "a {} event", "log")
180 (target: $target:expr, $($arg:tt)+) => ({
181 $crate::log!(target: $target, $crate::Level::Error, $($arg)+)
182 });
183
184 // error!("a {} event", "log")
185 ($($arg:tt)+) => ($crate::log!($crate::Level::Error, $($arg)+))
186}
187
188/// Logs a message at the warn level.
189///
190/// # Examples
191///
192/// ```
193/// use log::warn;
194///
195/// # let my_logger = log::__private_api::GlobalLogger;
196/// let warn_description = "Invalid Input";
197///
198/// warn!("Warning! {warn_description}!");
199/// warn!(target: "input_events", "App received warning: {warn_description}");
200/// warn!(logger: my_logger, "App received warning: {warn_description}");
201/// ```
202#[macro_export]
203#[clippy::format_args]
204macro_rules! warn {
205 // warn!(logger: my_logger, target: "my_target", key1 = 42, key2 = true; "a {} event", "log")
206 // warn!(logger: my_logger, target: "my_target", "a {} event", "log")
207 (logger: $logger:expr, target: $target:expr, $($arg:tt)+) => ({
208 $crate::log!(logger: $crate::__log_logger!($logger), target: $target, $crate::Level::Warn, $($arg)+)
209 });
210
211 // warn!(logger: my_logger, key1 = 42, key2 = true; "a {} event", "log")
212 // warn!(logger: my_logger, "a {} event", "log")
213 (logger: $logger:expr, $($arg:tt)+) => ({
214 $crate::log!(logger: $crate::__log_logger!($logger), $crate::Level::Warn, $($arg)+)
215 });
216
217 // warn!(target: "my_target", key1 = 42, key2 = true; "a {} event", "log")
218 // warn!(target: "my_target", "a {} event", "log")
219 (target: $target:expr, $($arg:tt)+) => ({
220 $crate::log!(target: $target, $crate::Level::Warn, $($arg)+)
221 });
222
223 // warn!("a {} event", "log")
224 ($($arg:tt)+) => ($crate::log!($crate::Level::Warn, $($arg)+))
225}
226
227/// Logs a message at the info level.
228///
229/// # Examples
230///
231/// ```
232/// use log::info;
233///
234/// # let my_logger = log::__private_api::GlobalLogger;
235/// # struct Connection { port: u32, speed: f32 }
236/// let conn_info = Connection { port: 40, speed: 3.20 };
237///
238/// info!("Connected to port {} at {} Mb/s", conn_info.port, conn_info.speed);
239/// info!(
240/// target: "connection_events",
241/// "Successful connection, port: {}, speed: {}",
242/// conn_info.port, conn_info.speed
243/// );
244/// info!(
245/// logger: my_logger,
246/// "Successful connection, port: {}, speed: {}",
247/// conn_info.port, conn_info.speed
248/// );
249/// ```
250#[macro_export]
251#[clippy::format_args]
252macro_rules! info {
253 // info!(logger: my_logger, target: "my_target", key1 = 42, key2 = true; "a {} event", "log")
254 // info!(logger: my_logger, target: "my_target", "a {} event", "log")
255 (logger: $logger:expr, target: $target:expr, $($arg:tt)+) => ({
256 $crate::log!(logger: $crate::__log_logger!($logger), target: $target, $crate::Level::Info, $($arg)+)
257 });
258
259 // info!(logger: my_logger, key1 = 42, key2 = true; "a {} event", "log")
260 // info!(logger: my_logger, "a {} event", "log")
261 (logger: $logger:expr, $($arg:tt)+) => ({
262 $crate::log!(logger: $crate::__log_logger!($logger), $crate::Level::Info, $($arg)+)
263 });
264
265 // info!(target: "my_target", key1 = 42, key2 = true; "a {} event", "log")
266 // info!(target: "my_target", "a {} event", "log")
267 (target: $target:expr, $($arg:tt)+) => ({
268 $crate::log!(target: $target, $crate::Level::Info, $($arg)+)
269 });
270
271 // info!("a {} event", "log")
272 ($($arg:tt)+) => ($crate::log!($crate::Level::Info, $($arg)+))
273}
274
275/// Logs a message at the debug level.
276///
277/// # Examples
278///
279/// ```
280/// use log::debug;
281///
282/// # let my_logger = log::__private_api::GlobalLogger;
283/// # struct Position { x: f32, y: f32 }
284/// let pos = Position { x: 3.234, y: -1.223 };
285///
286/// debug!("New position: x: {}, y: {}", pos.x, pos.y);
287/// debug!(target: "app_events", "New position: x: {}, y: {}", pos.x, pos.y);
288/// debug!(logger: my_logger, "New position: x: {}, y: {}", pos.x, pos.y);
289/// ```
290#[macro_export]
291#[clippy::format_args]
292macro_rules! debug {
293 // debug!(logger: my_logger, target: "my_target", key1 = 42, key2 = true; "a {} event", "log")
294 // debug!(logger: my_logger, target: "my_target", "a {} event", "log")
295 (logger: $logger:expr, target: $target:expr, $($arg:tt)+) => ({
296 $crate::log!(logger: $crate::__log_logger!($logger), target: $target, $crate::Level::Debug, $($arg)+)
297 });
298
299 // debug!(logger: my_logger, key1 = 42, key2 = true; "a {} event", "log")
300 // debug!(logger: my_logger, "a {} event", "log")
301 (logger: $logger:expr, $($arg:tt)+) => ({
302 $crate::log!(logger: $crate::__log_logger!($logger), $crate::Level::Debug, $($arg)+)
303 });
304
305 // debug!(target: "my_target", key1 = 42, key2 = true; "a {} event", "log")
306 // debug!(target: "my_target", "a {} event", "log")
307 (target: $target:expr, $($arg:tt)+) => ({
308 $crate::log!(target: $target, $crate::Level::Debug, $($arg)+)
309 });
310
311 // debug!("a {} event", "log")
312 ($($arg:tt)+) => ($crate::log!($crate::Level::Debug, $($arg)+))
313}
314
315/// Logs a message at the trace level.
316///
317/// # Examples
318///
319/// ```
320/// use log::trace;
321///
322/// # let my_logger = log::__private_api::GlobalLogger;
323/// # struct Position { x: f32, y: f32 }
324/// let pos = Position { x: 3.234, y: -1.223 };
325///
326/// trace!("Position is: x: {}, y: {}", pos.x, pos.y);
327/// trace!(target: "app_events", "x is {} and y is {}",
328/// if pos.x >= 0.0 { "positive" } else { "negative" },
329/// if pos.y >= 0.0 { "positive" } else { "negative" });
330/// trace!(logger: my_logger, "x is {} and y is {}",
331/// if pos.x >= 0.0 { "positive" } else { "negative" },
332/// if pos.y >= 0.0 { "positive" } else { "negative" });
333/// ```
334#[macro_export]
335#[clippy::format_args]
336macro_rules! trace {
337 // trace!(logger: my_logger, target: "my_target", key1 = 42, key2 = true; "a {} event", "log")
338 // trace!(logger: my_logger, target: "my_target", "a {} event", "log")
339 (logger: $logger:expr, target: $target:expr, $($arg:tt)+) => ({
340 $crate::log!(logger: $crate::__log_logger!($logger), target: $target, $crate::Level::Trace, $($arg)+)
341 });
342
343 // trace!(logger: my_logger, key1 = 42, key2 = true; "a {} event", "log")
344 // trace!(logger: my_logger, "a {} event", "log")
345 (logger: $logger:expr, $($arg:tt)+) => ({
346 $crate::log!(logger: $crate::__log_logger!($logger), $crate::Level::Trace, $($arg)+)
347 });
348
349 // trace!(target: "my_target", key1 = 42, key2 = true; "a {} event", "log")
350 // trace!(target: "my_target", "a {} event", "log")
351 (target: $target:expr, $($arg:tt)+) => ({
352 $crate::log!(target: $target, $crate::Level::Trace, $($arg)+)
353 });
354
355 // trace!("a {} event", "log")
356 ($($arg:tt)+) => ($crate::log!($crate::Level::Trace, $($arg)+))
357}
358
359/// Determines if a message logged at the specified level in that module will
360/// be logged.
361///
362/// This can be used to avoid expensive computation of log message arguments if
363/// the message would be ignored anyway.
364///
365/// # Examples
366///
367/// ```
368/// use log::{debug, log_enabled, Level};
369///
370/// # struct Data { x: u32, y: u32 }
371/// # fn expensive_call() -> Data { Data { x: 0, y: 0 } }
372/// # let my_logger = log::__private_api::GlobalLogger;
373/// if log_enabled!(Level::Debug) {
374/// let data = expensive_call();
375/// debug!("expensive debug data: {} {}", data.x, data.y);
376/// }
377///
378/// if log_enabled!(target: "Global", Level::Debug) {
379/// let data = expensive_call();
380/// debug!(target: "Global", "expensive debug data: {} {}", data.x, data.y);
381/// }
382///
383/// if log_enabled!(logger: my_logger, Level::Debug) {
384/// let data = expensive_call();
385/// debug!(target: "Global", "expensive debug data: {} {}", data.x, data.y);
386/// }
387/// ```
388///
389/// This macro accepts the same `target` and `logger` arguments as [`macro@log`].
390#[macro_export]
391macro_rules! log_enabled {
392 // log_enabled!(logger: my_logger, target: "my_target", Level::Info)
393 (logger: $logger:expr, target: $target:expr, $lvl:expr) => ({
394 $crate::__log_enabled!(logger: $crate::__log_logger!($logger), target: $target, $lvl)
395 });
396
397 // log_enabled!(logger: my_logger, Level::Info)
398 (logger: $logger:expr, $lvl:expr) => ({
399 $crate::__log_enabled!(logger: $crate::__log_logger!($logger), target: $crate::__private_api::module_path!(), $lvl)
400 });
401
402 // log_enabled!(target: "my_target", Level::Info)
403 (target: $target:expr, $lvl:expr) => ({
404 $crate::__log_enabled!(logger: $crate::__log_logger!(__log_global_logger), target: $target, $lvl)
405 });
406
407 // log_enabled!(Level::Info)
408 ($lvl:expr) => ({
409 $crate::__log_enabled!(logger: $crate::__log_logger!(__log_global_logger), target: $crate::__private_api::module_path!(), $lvl)
410 });
411}
412
413#[doc(hidden)]
414#[macro_export]
415macro_rules! __log_enabled {
416 // log_enabled!(logger: my_logger, target: "my_target", Level::Info)
417 (logger: $logger:expr, target: $target:expr, $lvl:expr) => {{
418 let lvl = $lvl;
419 lvl <= $crate::STATIC_MAX_LEVEL
420 && lvl <= $crate::max_level()
421 && $crate::__private_api::enabled($logger, lvl, $target)
422 }};
423}
424
425// Determine the logger to use, and whether to take it by-value or by reference
426
427#[doc(hidden)]
428#[macro_export]
429macro_rules! __log_logger {
430 (__log_global_logger) => {{
431 $crate::__private_api::GlobalLogger
432 }};
433
434 ($logger:expr) => {{
435 &($logger)
436 }};
437}
438
439// These macros use a pattern of #[cfg]s to produce nicer error
440// messages when log features aren't available
441
442#[doc(hidden)]
443#[macro_export]
444#[cfg(feature = "kv")]
445macro_rules! __log_key {
446 // key1 = 42
447 ($($args:ident)*) => {
448 $crate::__private_api::stringify!($($args)*)
449 };
450 // "key1" = 42
451 ($($args:expr)*) => {
452 $($args)*
453 };
454}
455
456#[doc(hidden)]
457#[macro_export]
458#[cfg(not(feature = "kv"))]
459macro_rules! __log_key {
460 ($($args:tt)*) => {
461 compile_error!("key value support requires the `kv` feature of `log`")
462 };
463}
464
465#[doc(hidden)]
466#[macro_export]
467#[cfg(feature = "kv")]
468macro_rules! __log_value {
469 // Entrypoint
470 ($key:tt = $args:expr) => {
471 $crate::__log_value!(($args):value)
472 };
473 ($key:tt :$capture:tt = $args:expr) => {
474 $crate::__log_value!(($args):$capture)
475 };
476 ($key:ident =) => {
477 $crate::__log_value!(($key):value)
478 };
479 ($key:ident :$capture:tt =) => {
480 $crate::__log_value!(($key):$capture)
481 };
482 // ToValue
483 (($args:expr):value) => {
484 $crate::__private_api::capture_to_value(&&$args)
485 };
486 // Debug
487 (($args:expr):?) => {
488 $crate::__private_api::capture_debug(&&$args)
489 };
490 (($args:expr):debug) => {
491 $crate::__private_api::capture_debug(&&$args)
492 };
493 // Display
494 (($args:expr):%) => {
495 $crate::__private_api::capture_display(&&$args)
496 };
497 (($args:expr):display) => {
498 $crate::__private_api::capture_display(&&$args)
499 };
500 //Error
501 (($args:expr):err) => {
502 $crate::__log_value_error!($args)
503 };
504 // sval::Value
505 (($args:expr):sval) => {
506 $crate::__log_value_sval!($args)
507 };
508 // serde::Serialize
509 (($args:expr):serde) => {
510 $crate::__log_value_serde!($args)
511 };
512}
513
514#[doc(hidden)]
515#[macro_export]
516#[cfg(not(feature = "kv"))]
517macro_rules! __log_value {
518 ($($args:tt)*) => {
519 compile_error!("key value support requires the `kv` feature of `log`")
520 };
521}
522
523#[doc(hidden)]
524#[macro_export]
525#[cfg(feature = "kv_sval")]
526macro_rules! __log_value_sval {
527 ($args:expr) => {
528 $crate::__private_api::capture_sval(&&$args)
529 };
530}
531
532#[doc(hidden)]
533#[macro_export]
534#[cfg(not(feature = "kv_sval"))]
535macro_rules! __log_value_sval {
536 ($args:expr) => {
537 compile_error!("capturing values as `sval::Value` requites the `kv_sval` feature of `log`")
538 };
539}
540
541#[doc(hidden)]
542#[macro_export]
543#[cfg(feature = "kv_serde")]
544macro_rules! __log_value_serde {
545 ($args:expr) => {
546 $crate::__private_api::capture_serde(&&$args)
547 };
548}
549
550#[doc(hidden)]
551#[macro_export]
552#[cfg(not(feature = "kv_serde"))]
553macro_rules! __log_value_serde {
554 ($args:expr) => {
555 compile_error!(
556 "capturing values as `serde::Serialize` requites the `kv_serde` feature of `log`"
557 )
558 };
559}
560
561#[doc(hidden)]
562#[macro_export]
563#[cfg(feature = "kv_std")]
564macro_rules! __log_value_error {
565 ($args:expr) => {
566 $crate::__private_api::capture_error(&$args)
567 };
568}
569
570#[doc(hidden)]
571#[macro_export]
572#[cfg(not(feature = "kv_std"))]
573macro_rules! __log_value_error {
574 ($args:expr) => {
575 compile_error!(
576 "capturing values as `std::error::Error` requites the `kv_std` feature of `log`"
577 )
578 };
579}