diff --git a/tracing/src/macros.rs b/tracing/src/macros.rs index b0791d63f9..21cb04c83e 100644 --- a/tracing/src/macros.rs +++ b/tracing/src/macros.rs @@ -131,6 +131,36 @@ macro_rules! span { }; } +/// Records multiple values on a span in a single call. As with recording +/// individual values, all fields must be declared when the span is created. +/// +/// This macro supports two optional sigils: +/// - `%` uses the Display implementation. +/// - `?` uses the Debug implementation. +/// +/// For more details, see the [top-level documentation][lib]. +/// +/// [lib]: tracing/#recording-fields +/// +/// # Examples +/// +/// ``` +/// # use tracing::{field, info_span, record_all}; +/// let span = info_span!("my span", field1 = field::Empty, field2 = field::Empty, field3 = field::Empty).entered(); +/// record_all!(span, field1 = ?"1", field2 = %"2", field3 = 3); +/// ``` +#[macro_export] +macro_rules! record_all { + ($span:expr, $($fields:tt)*) => { + if let Some(meta) = $span.metadata() { + $span.record_all(&$crate::valueset!( + meta.fields(), + $($fields)* + )); + } + }; +} + /// Constructs a span at the trace level. /// /// [Fields] and [attributes] are set using the same syntax as the [`span!`] diff --git a/tracing/src/span.rs b/tracing/src/span.rs index daa4a9ae5e..cff4d21ba5 100644 --- a/tracing/src/span.rs +++ b/tracing/src/span.rs @@ -1191,6 +1191,11 @@ impl Span { /// span.record("parting", "you will be remembered"); /// ``` /// + ///
+ ///
+    /// **Note**: To record several values in just one call, see the [`record_all!`](crate::record_all!) macro.
+    /// 
+ /// /// [`field::Empty`]: super::field::Empty /// [`Metadata`]: super::Metadata pub fn record(&self, field: &Q, value: V) -> &Self @@ -1212,6 +1217,7 @@ impl Span { } /// Records all the fields in the provided `ValueSet`. + #[doc(hidden)] pub fn record_all(&self, values: &field::ValueSet<'_>) -> &Self { let record = Record::new(values); if let Some(ref inner) = self.inner { diff --git a/tracing/tests/span.rs b/tracing/tests/span.rs index bc0798f162..f2b5e5eccd 100644 --- a/tracing/tests/span.rs +++ b/tracing/tests/span.rs @@ -8,8 +8,8 @@ use std::thread; use tracing::{ collect::with_default, error_span, - field::{debug, display}, - Level, Span, + field::{debug, display, Empty}, + record_all, Level, Span, }; use tracing_mock::*; @@ -612,6 +612,50 @@ fn record_new_values_for_fields() { handle.assert_finished(); } +/// Tests record_all! macro, which is a wrapper for Span.record_all(). +/// Placed here instead of tests/macros.rs, because it uses tracing_mock, which +/// requires std lib. Other macro tests exclude std lib to verify the macros do +/// not dependend on it. +#[cfg_attr(target_arch = "wasm32", wasm_bindgen_test::wasm_bindgen_test)] +#[test] +fn record_all_macro_records_new_values_for_fields() { + let (collector, handle) = collector::mock() + .new_span( + expect::span() + .named("foo") + .with_fields(expect::field("bar")), + ) + .record( + expect::span().named("foo"), + expect::field("bar") + .with_value(&5) + .and(expect::field("baz").with_value(&"BAZ")) + .and(expect::field("qux").with_value(&display("qux"))) + .and(expect::field("quux").with_value(&debug("QuuX"))) + .only(), + ) + .enter(expect::span().named("foo")) + .exit(expect::span().named("foo")) + .drop_span(expect::span().named("foo")) + .only() + .run_with_handle(); + + with_default(collector, || { + let span = tracing::span!( + Level::TRACE, + "foo", + bar = 1, + baz = 2, + qux = Empty, + quux = Empty + ); + record_all!(span, bar = 5, baz = "BAZ", qux = %"qux", quux = ?"QuuX"); + span.in_scope(|| {}) + }); + + handle.assert_finished(); +} + #[cfg_attr(target_arch = "wasm32", wasm_bindgen_test::wasm_bindgen_test)] #[test] fn new_span_with_target_and_log_level() {