diff --git a/spdlog-macros/src/lib.rs b/spdlog-macros/src/lib.rs index c9d60fd6..53c7814a 100644 --- a/spdlog-macros/src/lib.rs +++ b/spdlog-macros/src/lib.rs @@ -5,16 +5,17 @@ //! //! [`spdlog-rs`]: https://crates.io/crates/spdlog-rs +mod normalize_forward; mod pattern; use proc_macro::TokenStream; use proc_macro2::TokenStream as TokenStream2; -use spdlog_internal::pattern_parser::Result; +use quote::quote; #[proc_macro] pub fn pattern(input: TokenStream) -> TokenStream { let pattern = syn::parse_macro_input!(input); - into_or_error(pattern::pattern_impl(pattern)) + into_or_error(pattern::pattern_impl(pattern).map_err(Error::PatternParser)) } #[proc_macro] @@ -23,7 +24,7 @@ pub fn runtime_pattern(input: TokenStream) -> TokenStream { // token which is used in the custom patterns. let runtime_pattern = syn::parse_macro_input!(input); - into_or_error(pattern::runtime_pattern_impl(runtime_pattern)) + into_or_error(pattern::runtime_pattern_impl(runtime_pattern).map_err(Error::PatternParser)) } #[proc_macro] @@ -33,9 +34,39 @@ pub fn runtime_pattern_disabled(_: TokenStream) -> TokenStream { ); } -fn into_or_error(result: Result) -> TokenStream { +// Example: +// +// ```rust +// normalize_forward!(callback => default[opt1: 1, opt2: {}, opt3: { 3 }, d], opt1: 10, a, b, c, opt3: { 30 }); +// // will be converted to +// spdlog::callback!(opt1: 10, opt2: {}, opt3: { 30 }, d, a, b, c); +// ``` +#[proc_macro] +pub fn normalize_forward(input: TokenStream) -> TokenStream { + let normalize = syn::parse_macro_input!(input); + into_or_error(normalize_forward::normalize(normalize).map_err(Error::NormalizeForward)) +} + +enum Error { + PatternParser(spdlog_internal::pattern_parser::Error), + NormalizeForward(syn::Error), +} + +impl Error { + fn emit(self) -> TokenStream2 { + match self { + Error::PatternParser(err) => { + let error = err.to_string(); + quote!(compile_error!(#error)) + } + Error::NormalizeForward(err) => err.to_compile_error(), + } + } +} + +fn into_or_error(result: Result) -> TokenStream { match result { Ok(stream) => stream.into(), - Err(err) => panic!("{}", err), + Err(err) => err.emit().into(), } } diff --git a/spdlog-macros/src/normalize_forward.rs b/spdlog-macros/src/normalize_forward.rs new file mode 100644 index 00000000..e71e8da4 --- /dev/null +++ b/spdlog-macros/src/normalize_forward.rs @@ -0,0 +1,238 @@ +use std::collections::HashSet; + +use proc_macro2::TokenStream; +use quote::{quote, ToTokens}; +use syn::{ + braced, bracketed, + parse::{discouraged::Speculative, Parse, ParseStream}, + Expr, Ident, Token, +}; + +pub struct Normalize { + callback: Ident, + default_list: DefaultArgsList, + args: Args, +} + +impl Parse for Normalize { + fn parse(input: ParseStream) -> syn::Result { + let callback = input.parse::()?; + input.parse::]>()?; + let default_list = DefaultArgsList::parse(input)?; + input.parse::()?; + let args = Args::parse(input)?; + Ok(Self { + callback, + default_list, + args, + }) + } +} + +struct DefaultArgsList(Vec); + +impl Parse for DefaultArgsList { + fn parse(input: ParseStream) -> syn::Result { + input.parse::()?; + let list; + bracketed!(list in input); + let list = list + .parse_terminated(Arg::parse, Token![,])? + .into_iter() + .collect(); + Ok(Self(list)) + } +} + +struct Args(Vec); + +impl Parse for Args { + fn parse(input: ParseStream) -> syn::Result { + let args = input.parse_terminated(Arg::parse, Token![,])?; + Ok(Self(args.into_iter().collect())) + } +} + +enum Arg { + Optional(OptionalArg), + Other(ArgValue), +} + +impl Arg { + fn as_optional(&self) -> Option<&OptionalArg> { + match self { + Self::Optional(arg) => Some(arg), + _ => None, + } + } + + fn as_optional_mut(&mut self) -> Option<&mut OptionalArg> { + match self { + Self::Optional(arg) => Some(arg), + _ => None, + } + } +} + +impl Parse for Arg { + fn parse(input: ParseStream) -> syn::Result { + let fork = input.fork(); + match OptionalArg::parse(&fork) { + Ok(opt_arg) => { + input.advance_to(&fork); + Ok(Self::Optional(opt_arg)) + } + Err(_) => Ok(Self::Other(ArgValue::parse(input)?)), + } + } +} + +struct OptionalArg { + name: Ident, + value: ArgValue, +} + +impl Parse for OptionalArg { + fn parse(input: ParseStream) -> syn::Result { + let name = input.parse::()?; + input.parse::()?; + let value = ArgValue::parse(input)?; + Ok(Self { name, value }) + } +} + +enum ArgValue { + Expr(Expr), + Braced(BraceAny), +} + +impl ArgValue { + fn into_token_stream(self) -> TokenStream { + match self { + Self::Expr(expr) => expr.into_token_stream(), + Self::Braced(braced) => braced.0, + } + } +} + +impl Parse for ArgValue { + fn parse(input: ParseStream) -> syn::Result { + let fork = input.fork(); + + match Expr::parse(&fork) { + Ok(expr) => { + input.advance_to(&fork); + Ok(Self::Expr(expr)) + } + Err(_) => Ok(BraceAny::parse(input).map(Self::Braced)?), + } + } +} + +struct BraceAny(TokenStream); + +impl Parse for BraceAny { + fn parse(input: ParseStream) -> syn::Result { + let content; + braced!(content in input); + let ts: TokenStream = content.parse()?; + Ok(Self(quote!({#ts}))) + } +} + +fn check_inputs(normalize: &Normalize) -> syn::Result<()> { + let mut seen_keys = HashSet::new(); + for arg in normalize.args.0.iter().filter_map(|arg| arg.as_optional()) { + if !seen_keys.insert(&arg.name) { + return Err(syn::Error::new( + arg.name.span(), + format!("found duplicate optional argument '{}'", arg.name), + )); + } + } + + let groups = normalize + .args + .0 + .split_inclusive(|arg| matches!(arg, Arg::Optional(_))) + .filter(|group| { + group + .iter() + .find(|arg| !matches!(arg, Arg::Optional(_))) + .is_some() + }) + .collect::>(); + if groups.len() > 1 { + return Err(syn::Error::new( + groups + .first() + .and_then(|group| group.last()) + .and_then(|arg| arg.as_optional()) + .unwrap() + .name + .span(), + "optional arguments cannot occur in the middle of regular arguments", + )); + } + + Ok(()) +} + +pub fn normalize(normalize: Normalize) -> syn::Result { + check_inputs(&normalize)?; + + let mut default_args = normalize.default_list.0; + let mut other_args = vec![]; + + for input_arg in normalize.args.0 { + match input_arg { + Arg::Optional(input_arg) => { + let stored = default_args + .iter_mut() + .find_map(|allowed| { + allowed + .as_optional_mut() + .and_then(|allowed| (allowed.name == input_arg.name).then_some(allowed)) + }) + .ok_or_else(|| { + syn::Error::new( + input_arg.name.span(), + format!("unknown optional parameter '{}'", input_arg.name), + ) + })?; + stored.value = input_arg.value; + } + Arg::Other(input_arg) => { + other_args.push(input_arg); + } + } + } + + let callback = normalize.callback; + let default_args = default_args + .into_iter() + .map(|arg| match arg { + Arg::Optional(arg) => { + let name = arg.name; + let value = arg.value.into_token_stream(); + quote!(#name: #value) + } + Arg::Other(arg) => { + let value = arg.into_token_stream(); + quote!(#value) + } + }) + .collect::>(); + let other_args = other_args + .into_iter() + .map(|arg| { + let ts = arg.into_token_stream(); + quote!(#ts) + }) + .collect::>(); + + let emitted = quote! { + ::spdlog::#callback!(#(#default_args),*, #(#other_args),*) + }; + Ok(emitted) +} diff --git a/spdlog/src/lib.rs b/spdlog/src/lib.rs index ed5d4202..ebac34cb 100644 --- a/spdlog/src/lib.rs +++ b/spdlog/src/lib.rs @@ -278,6 +278,10 @@ #![cfg_attr(all(doc, CHANNEL_NIGHTLY), feature(doc_auto_cfg))] #![warn(missing_docs)] +// Used for referencing from proc-macros +// Credits: https://stackoverflow.com/a/57049687 +extern crate self as spdlog; + mod env_level; pub mod error; pub mod formatter; @@ -308,6 +312,8 @@ pub use log_crate_proxy::*; pub use logger::*; pub use record::*; pub use source_location::*; +#[doc(hidden)] +pub use spdlog_macros::normalize_forward as __normalize_forward; pub use string_buf::StringBuf; #[cfg(feature = "multi-thread")] pub use thread_pool::*; diff --git a/spdlog/src/log_macros.rs b/spdlog/src/log_macros.rs index 05f92e82..9fc558b6 100644 --- a/spdlog/src/log_macros.rs +++ b/spdlog/src/log_macros.rs @@ -21,6 +21,14 @@ /// [`Level`]: crate::Level #[macro_export] macro_rules! log { + ($($input:tt)+) => { + $crate::__normalize_forward!(__log_impl => default[logger: $crate::default_logger(), kv: {}], $($input)+) + }; +} + +#[doc(hidden)] +#[macro_export] +macro_rules! __log_impl { (logger: $logger:expr, kv: $kv:tt, $level:expr, $($arg:tt)+) => ({ let logger = &$logger; const LEVEL: $crate::Level = $level; @@ -29,9 +37,6 @@ macro_rules! log { $crate::__log(logger, LEVEL, $crate::source_location_current!(), $crate::__kv!($kv), format_args!($($arg)+)); } }); - (logger: $logger:expr, $level:expr, $($arg:tt)+) => ($crate::log!(logger: $logger, kv: {}, $level, $($arg)+)); - (kv: $kv:tt, $level:expr, $($arg:tt)+) => ($crate::log!(logger: $crate::default_logger(), kv: $kv, $level, $($arg)+)); - ($level:expr, $($arg:tt)+) => ($crate::log!(logger: $crate::default_logger(), kv: {}, $level, $($arg)+)); } /// Logs a message at the critical level. @@ -52,18 +57,9 @@ macro_rules! log { /// ``` #[macro_export] macro_rules! critical { - (logger: $logger:expr, kv: $kv:tt, $($arg:tt)+) => ( - $crate::log!(logger: $logger, kv: $kv, $crate::Level::Critical, $($arg)+) - ); - (logger: $logger:expr, $($arg:tt)+) => ( - $crate::log!(logger: $logger, $crate::Level::Critical, $($arg)+) - ); - (kv: $kv:tt, $($arg:tt)+) => ( - $crate::log!(kv: $kv, $crate::Level::Critical, $($arg)+) - ); - ($($arg:tt)+) => ( - $crate::log!($crate::Level::Critical, $($arg)+) - ) + ($($input:tt)+) => { + $crate::__normalize_forward!(__log_impl => default[logger: $crate::default_logger(), kv: {}, $crate::Level::Critical], $($input)+) + }; } /// Logs a message at the error level. @@ -84,18 +80,9 @@ macro_rules! critical { /// ``` #[macro_export] macro_rules! error { - (logger: $logger:expr, kv: $kv:tt, $($arg:tt)+) => ( - $crate::log!(logger: $logger, kv: $kv, $crate::Level::Error, $($arg)+) - ); - (logger: $logger:expr, $($arg:tt)+) => ( - $crate::log!(logger: $logger, $crate::Level::Error, $($arg)+) - ); - (kv: $kv:tt, $($arg:tt)+) => ( - $crate::log!(kv: $kv, $crate::Level::Error, $($arg)+) - ); - ($($arg:tt)+) => ( - $crate::log!($crate::Level::Error, $($arg)+) - ) + ($($input:tt)+) => { + $crate::__normalize_forward!(__log_impl => default[logger: $crate::default_logger(), kv: {}, $crate::Level::Error], $($input)+) + }; } /// Logs a message at the warn level. @@ -116,18 +103,9 @@ macro_rules! error { /// ``` #[macro_export] macro_rules! warn { - (logger: $logger:expr, kv: $kv:tt, $($arg:tt)+) => ( - $crate::log!(logger: $logger, kv: $kv, $crate::Level::Warn, $($arg)+) - ); - (logger: $logger:expr, $($arg:tt)+) => ( - $crate::log!(logger: $logger, $crate::Level::Warn, $($arg)+) - ); - (kv: $kv:tt, $($arg:tt)+) => ( - $crate::log!(kv: $kv, $crate::Level::Warn, $($arg)+) - ); - ($($arg:tt)+) => ( - $crate::log!($crate::Level::Warn, $($arg)+) - ) + ($($input:tt)+) => { + $crate::__normalize_forward!(__log_impl => default[logger: $crate::default_logger(), kv: {}, $crate::Level::Warn], $($input)+) + }; } /// Logs a message at the info level. @@ -149,18 +127,9 @@ macro_rules! warn { /// ``` #[macro_export] macro_rules! info { - (logger: $logger:expr, kv: $kv:tt, $($arg:tt)+) => ( - $crate::log!(logger: $logger, kv: $kv, $crate::Level::Info, $($arg)+) - ); - (logger: $logger:expr, $($arg:tt)+) => ( - $crate::log!(logger: $logger, $crate::Level::Info, $($arg)+) - ); - (kv: $kv:tt, $($arg:tt)+) => ( - $crate::log!(kv: $kv, $crate::Level::Info, $($arg)+) - ); - ($($arg:tt)+) => ( - $crate::log!($crate::Level::Info, $($arg)+) - ) + ($($input:tt)+) => { + $crate::__normalize_forward!(__log_impl => default[logger: $crate::default_logger(), kv: {}, $crate::Level::Info], $($input)+) + }; } /// Logs a message at the debug level. @@ -182,18 +151,9 @@ macro_rules! info { /// ``` #[macro_export] macro_rules! debug { - (logger: $logger:expr, kv: $kv:tt, $($arg:tt)+) => ( - $crate::log!(logger: $logger, kv: $kv, $crate::Level::Debug, $($arg)+) - ); - (logger: $logger:expr, $($arg:tt)+) => ( - $crate::log!(logger: $logger, $crate::Level::Debug, $($arg)+) - ); - (kv: $kv:tt, $($arg:tt)+) => ( - $crate::log!(kv: $kv, $crate::Level::Debug, $($arg)+) - ); - ($($arg:tt)+) => ( - $crate::log!($crate::Level::Debug, $($arg)+) - ) + ($($input:tt)+) => { + $crate::__normalize_forward!(__log_impl => default[logger: $crate::default_logger(), kv: {}, $crate::Level::Debug], $($input)+) + }; } /// Logs a message at the trace level. @@ -217,18 +177,9 @@ macro_rules! debug { /// ``` #[macro_export] macro_rules! trace { - (logger: $logger:expr, kv: $kv:tt, $($arg:tt)+) => ( - $crate::log!(logger: $logger, kv: $kv, $crate::Level::Trace, $($arg)+) - ); - (logger: $logger:expr, $($arg:tt)+) => ( - $crate::log!(logger: $logger, $crate::Level::Trace, $($arg)+) - ); - (kv: $kv:tt, $($arg:tt)+) => ( - $crate::log!(kv: $kv, $crate::Level::Trace, $($arg)+) - ); - ($($arg:tt)+) => ( - $crate::log!($crate::Level::Trace, $($arg)+) - ) + ($($input:tt)+) => { + $crate::__normalize_forward!(__log_impl => default[logger: $crate::default_logger(), kv: {}, $crate::Level::Trace], $($input)+) + }; } #[doc(hidden)] @@ -320,6 +271,16 @@ mod tests { Level::Info, "logger, kv(s,mod,vref)".to_string(), ), + ( + vec![(KeyInner::StaticStr("mod_di"), "display".to_string())], + Level::Debug, + "arbitrary order = logger, fmt, kv".to_string(), + ), + ( + vec![(KeyInner::StaticStr("mod_di"), "display".to_string())], + Level::Debug, + "arbitrary order = fmt, logger, kv".to_string(), + ), ]; log!(logger: test, Level::Info, "logger"); @@ -331,6 +292,8 @@ mod tests { log!(logger: test, kv: { n }, Level::Trace, "logger, kv(1,vref)"); log!(logger: test, kv: { mod_di: }, Level::Debug, "logger, kv(mod,vref)"); log!(logger: test, kv: { n, mod_di:, mod_de:? }, Level::Info, "logger, kv(s,mod,vref)"); + log!(logger: test, Level::Debug, "arbitrary order = logger, fmt, kv", kv: { mod_di: }); + log!(Level::Debug, "arbitrary order = fmt, logger, kv", logger: test, kv: { mod_di: }); macro_rules! add_records { ( $($level:ident => $variant:ident),+ ) => { @@ -383,6 +346,20 @@ mod tests { Level::$variant, format!("{}: logger, kv(s,mod,vref)", stringify!($level)) )); + + $level!(logger: test, "{}: arbitrary order = logger, fmt, kv", stringify!($level), kv: { mod_di: }); + check.push(( + vec![(KeyInner::StaticStr("mod_di"), "display".to_string())], + Level::$variant, + format!("{}: arbitrary order = logger, fmt, kv", stringify!($level)) + )); + + $level!("{}: arbitrary order = fmt, logger, kv", stringify!($level), logger: test, kv: { mod_di: }); + check.push(( + vec![(KeyInner::StaticStr("mod_di"), "display".to_string())], + Level::$variant, + format!("{}: arbitrary order = fmt, logger, kv", stringify!($level)) + )); )+ }; } diff --git a/spdlog/tests/compile_fail.rs b/spdlog/tests/compile_fail.rs index 17c691e4..17c5204e 100644 --- a/spdlog/tests/compile_fail.rs +++ b/spdlog/tests/compile_fail.rs @@ -2,6 +2,7 @@ fn compile_fail() { let t = trybuild::TestCases::new(); + t.compile_fail("tests/compile_fail/logging_macro_*.rs"); t.compile_fail("tests/compile_fail/pattern_macro_*.rs"); #[cfg(feature = "runtime-pattern")] t.compile_fail("tests/compile_fail/pattern_runtime_macro_*.rs"); diff --git a/spdlog/tests/compile_fail/logging_macro_syntax.rs b/spdlog/tests/compile_fail/logging_macro_syntax.rs new file mode 100644 index 00000000..81f0c0b7 --- /dev/null +++ b/spdlog/tests/compile_fail/logging_macro_syntax.rs @@ -0,0 +1,11 @@ +use spdlog::prelude::*; + +fn optional_args() { + let logger = spdlog::default_logger(); + + log!(unknown: 1, Level::Info, "unknown optional arg"); + log!(Level::Info, logger: logger, "optional arg in the middle"); + log!(logger: logger, Level::Info, "duplicate optional args", logger: logger); +} + +fn main() {} diff --git a/spdlog/tests/compile_fail/logging_macro_syntax.stderr b/spdlog/tests/compile_fail/logging_macro_syntax.stderr new file mode 100644 index 00000000..e1ddaa62 --- /dev/null +++ b/spdlog/tests/compile_fail/logging_macro_syntax.stderr @@ -0,0 +1,17 @@ +error: unknown optional parameter 'unknown' + --> tests/compile_fail/logging_macro_syntax.rs:6:10 + | +6 | log!(unknown: 1, Level::Info, "unknown optional arg"); + | ^^^^^^^ + +error: optional arguments cannot occur in the middle of regular arguments + --> tests/compile_fail/logging_macro_syntax.rs:7:23 + | +7 | log!(Level::Info, logger: logger, "optional arg in the middle"); + | ^^^^^^ + +error: found duplicate optional argument 'logger' + --> tests/compile_fail/logging_macro_syntax.rs:8:66 + | +8 | log!(logger: logger, Level::Info, "duplicate optional args", logger: logger); + | ^^^^^^