Skip to content

Commit d609dc0

Browse files
committed
macros: Remove 'r#' prefix from raw identifiers in field names
1 parent 2297bbb commit d609dc0

File tree

3 files changed

+69
-6
lines changed

3 files changed

+69
-6
lines changed

tracing/src/lib.rs

+62-2
Original file line numberDiff line numberDiff line change
@@ -1002,12 +1002,12 @@ pub mod span;
10021002
pub mod __macro_support {
10031003
pub use crate::callsite::{Callsite, Registration};
10041004
use crate::{collect::Interest, Metadata};
1005-
use core::fmt;
10061005
use core::sync::atomic::{AtomicU8, Ordering};
1006+
use core::{fmt, mem};
10071007
// Re-export the `core` functions that are used in macros. This allows
10081008
// a crate to be named `core` and avoid name clashes.
10091009
// See here: https://github.com/tokio-rs/tracing/issues/2761
1010-
pub use core::{concat, file, format_args, iter::Iterator, line, option::Option};
1010+
pub use core::{concat, file, format_args, iter::Iterator, line, option::Option, stringify};
10111011

10121012
/// Callsite implementation used by macro-generated code.
10131013
///
@@ -1195,6 +1195,66 @@ pub mod __macro_support {
11951195
.finish()
11961196
}
11971197
}
1198+
1199+
/// Implementation detail used for constructing FieldSet names from raw
1200+
/// identifiers. In `info!(..., r#type = "...")` the macro would end up
1201+
/// constructing a name equivalent to `FieldName(*b"type")`.
1202+
pub struct FieldName<const N: usize>([u8; N]);
1203+
1204+
impl<const N: usize> FieldName<N> {
1205+
/// Convert `"prefix.r#keyword.suffix"` to `b"prefix.keyword.suffix"`.
1206+
pub const fn new(input: &str) -> Self {
1207+
let input = input.as_bytes();
1208+
let mut output = [0u8; N];
1209+
let mut read = 0;
1210+
let mut write = 0;
1211+
while read < input.len() {
1212+
if read + 1 < input.len() && input[read] == b'r' && input[read + 1] == b'#' {
1213+
read += 2;
1214+
}
1215+
output[write] = input[read];
1216+
read += 1;
1217+
write += 1;
1218+
}
1219+
assert!(write == N);
1220+
Self(output)
1221+
}
1222+
1223+
pub const fn as_str(&self) -> &str {
1224+
// SAFETY: Because of the private visibility of self.0, it must have
1225+
// been computed by Self::new. So these bytes are all of the bytes
1226+
// of some original valid UTF-8 string, but with "r#" substrings
1227+
// removed, which cannot have produced invalid UTF-8.
1228+
unsafe { mem::transmute(self.0.as_slice()) }
1229+
}
1230+
}
1231+
1232+
impl FieldName<0> {
1233+
/// For `"prefix.r#keyword.suffix"` compute `"prefix.keyword.suffix".len()`.
1234+
pub const fn len(input: &str) -> usize {
1235+
// Count occurrences of "r#"
1236+
let mut raw = 0;
1237+
1238+
let mut i = 0;
1239+
while i < input.len() {
1240+
if input.as_bytes()[i] == b'#' {
1241+
raw += 1;
1242+
}
1243+
i += 1;
1244+
}
1245+
1246+
input.len() - 2 * raw
1247+
}
1248+
}
1249+
1250+
impl<const N: usize> fmt::Debug for FieldName<N> {
1251+
fn fmt(&self, formatter: &mut fmt::Formatter<'_>) -> fmt::Result {
1252+
formatter
1253+
.debug_tuple("FieldName")
1254+
.field(&self.as_str())
1255+
.finish()
1256+
}
1257+
}
11981258
}
11991259

12001260
#[cfg(feature = "log")]

tracing/src/macros.rs

+6-3
Original file line numberDiff line numberDiff line change
@@ -3069,9 +3069,12 @@ macro_rules! level_to_log {
30693069
#[doc(hidden)]
30703070
#[macro_export]
30713071
macro_rules! __tracing_stringify {
3072-
($($t:tt)*) => {
3073-
stringify!($($t)*)
3074-
};
3072+
($($k:ident).+) => {{
3073+
const NAME: $crate::__macro_support::FieldName<{
3074+
$crate::__macro_support::FieldName::len($crate::__macro_support::stringify!($($k).+))
3075+
}> = $crate::__macro_support::FieldName::new($crate::__macro_support::stringify!($($k).+));
3076+
NAME.as_str()
3077+
}};
30753078
}
30763079

30773080
#[cfg(not(feature = "log"))]

tracing/tests/event.rs

+1-1
Original file line numberDiff line numberDiff line change
@@ -514,7 +514,7 @@ fn keyword_ident_in_field_name() {
514514
#[test]
515515
fn raw_ident_in_field_name() {
516516
let (collector, handle) = collector::mock()
517-
.event(expect::event().with_fields(expect::field("this.r#type").with_value(&"Value")))
517+
.event(expect::event().with_fields(expect::field("this.type").with_value(&"Value")))
518518
.only()
519519
.run_with_handle();
520520

0 commit comments

Comments
 (0)