🛈 Note: This is pre-release documentation for the upcoming tracing 0.2.0 ecosystem.

For the release documentation, please see docs.rs, instead.

tracing_core/
span.rs

1//! Spans represent periods of time in the execution of a program.
2use core::num::NonZeroU64;
3
4use crate::field::FieldSet;
5use crate::parent::Parent;
6use crate::{field, Metadata};
7
8/// Identifies a span within the context of a collector.
9///
10/// They are generated by [collector]s for each span as it is created, by
11/// the [`new_span`] trait method. See the documentation for that method for
12/// more information on span ID generation.
13///
14/// [collector]: super::collect::Collect
15/// [`new_span`]: super::collect::Collect::new_span
16#[derive(Clone, Debug, PartialEq, Eq, Hash)]
17pub struct Id(NonZeroU64);
18
19/// Attributes provided to a collector describing a new span when it is
20/// created.
21#[derive(Debug)]
22pub struct Attributes<'a> {
23    metadata: &'static Metadata<'static>,
24    values: &'a field::ValueSet<'a>,
25    parent: Parent,
26}
27
28/// A set of fields recorded by a span.
29#[derive(Debug)]
30pub struct Record<'a> {
31    values: &'a field::ValueSet<'a>,
32}
33
34/// Indicates what [the collector considers] the "current" span.
35///
36/// As collectors may not track a notion of a current span, this has three
37/// possible states:
38/// - "unknown", indicating that the collector does not track a current span,
39/// - "none", indicating that the current context is known to not be in a span,
40/// - "some", with the current span's [`Id`] and [`Metadata`].
41///
42/// [the collector considers]: super::collect::Collect::current_span
43/// [`Metadata`]: super::metadata::Metadata
44#[derive(Debug)]
45pub struct Current {
46    inner: CurrentInner,
47}
48
49#[derive(Debug)]
50enum CurrentInner {
51    Current {
52        id: Id,
53        metadata: &'static Metadata<'static>,
54    },
55    None,
56    Unknown,
57}
58
59// ===== impl Span =====
60
61impl Id {
62    /// Constructs a new span ID from the given `u64`.
63    ///
64    /// <div class="example-wrap" style="display:inline-block">
65    /// <pre class="ignore" style="white-space:normal;font:inherit;">
66    /// <strong>Note</strong>: Span IDs must be greater than zero.</pre></div>
67    ///
68    /// # Panics
69    /// - If the provided `u64` is 0.
70    pub fn from_u64(u: u64) -> Self {
71        Id(NonZeroU64::new(u).expect("span IDs must be > 0"))
72    }
73
74    /// Constructs a new span ID from the given `NonZeroU64`.
75    ///
76    /// Unlike [`Id::from_u64`](Self::from_u64), this will never panic.
77    #[inline]
78    pub const fn from_non_zero_u64(id: NonZeroU64) -> Self {
79        Id(id)
80    }
81
82    // Allow `into` by-ref since we don't want to impl Copy for Id
83    #[allow(clippy::wrong_self_convention)]
84    /// Returns the span's ID as a `u64`.
85    pub fn into_u64(&self) -> u64 {
86        self.0.get()
87    }
88
89    // Allow `into` by-ref since we don't want to impl Copy for Id
90    #[allow(clippy::wrong_self_convention)]
91    /// Returns the span's ID as a `NonZeroU64`.
92    #[inline]
93    pub const fn into_non_zero_u64(&self) -> NonZeroU64 {
94        self.0
95    }
96}
97
98impl<'a> From<&'a Id> for Option<Id> {
99    fn from(id: &'a Id) -> Self {
100        Some(id.clone())
101    }
102}
103
104// ===== impl Attributes =====
105
106impl<'a> Attributes<'a> {
107    /// Returns `Attributes` describing a new child span of the current span,
108    /// with the provided metadata and values.
109    pub fn new(metadata: &'static Metadata<'static>, values: &'a field::ValueSet<'a>) -> Self {
110        Attributes {
111            metadata,
112            values,
113            parent: Parent::Current,
114        }
115    }
116
117    /// Returns `Attributes` describing a new span at the root of its own trace
118    /// tree, with the provided metadata and values.
119    pub fn new_root(metadata: &'static Metadata<'static>, values: &'a field::ValueSet<'a>) -> Self {
120        Attributes {
121            metadata,
122            values,
123            parent: Parent::Root,
124        }
125    }
126
127    /// Returns `Attributes` describing a new child span of the specified
128    /// parent span, with the provided metadata and values.
129    pub fn child_of(
130        parent: Id,
131        metadata: &'static Metadata<'static>,
132        values: &'a field::ValueSet<'a>,
133    ) -> Self {
134        Attributes {
135            metadata,
136            values,
137            parent: Parent::Explicit(parent),
138        }
139    }
140
141    /// Returns a reference to the new span's metadata.
142    pub fn metadata(&self) -> &'static Metadata<'static> {
143        self.metadata
144    }
145
146    /// Returns a reference to a `ValueSet` containing any values the new span
147    /// was created with.
148    pub fn values(&self) -> &field::ValueSet<'a> {
149        self.values
150    }
151
152    /// Returns true if the new span should be a root.
153    pub fn is_root(&self) -> bool {
154        matches!(self.parent, Parent::Root)
155    }
156
157    /// Returns true if the new span's parent should be determined based on the
158    /// current context.
159    ///
160    /// If this is true and the current thread is currently inside a span, then
161    /// that span should be the new span's parent. Otherwise, if the current
162    /// thread is _not_ inside a span, then the new span will be the root of its
163    /// own trace tree.
164    pub fn is_contextual(&self) -> bool {
165        matches!(self.parent, Parent::Current)
166    }
167
168    /// Returns the new span's explicitly-specified parent, if there is one.
169    ///
170    /// Otherwise (if the new span is a root or is a child of the current span),
171    /// returns `None`.
172    pub fn parent(&self) -> Option<&Id> {
173        match self.parent {
174            Parent::Explicit(ref p) => Some(p),
175            _ => None,
176        }
177    }
178
179    /// Records all the fields in this set of `Attributes` with the provided
180    /// [Visitor].
181    ///
182    /// [visitor]: super::field::Visit
183    pub fn record(&self, visitor: &mut dyn field::Visit) {
184        self.values.record(visitor)
185    }
186
187    /// Returns `true` if this set of `Attributes` contains a value for the
188    /// given `Field`.
189    pub fn contains(&self, field: &field::Field) -> bool {
190        self.values.contains(field)
191    }
192
193    /// Returns true if this set of `Attributes` contains _no_ values.
194    pub fn is_empty(&self) -> bool {
195        self.values.is_empty()
196    }
197
198    /// Returns the set of all [fields] defined by this span's [`Metadata`].
199    ///
200    /// Note that the [`FieldSet`] returned by this method includes *all* the
201    /// fields declared by this span, not just those with values that are recorded
202    /// as part of this set of `Attributes`. Other fields with values not present in
203    /// this `Attributes`' value set may [record] values later.
204    ///
205    /// [fields]: crate::field
206    /// [record]: Attributes::record()
207    /// [`Metadata`]: crate::metadata::Metadata
208    /// [`FieldSet`]: crate::field::FieldSet
209    pub fn fields(&self) -> &FieldSet {
210        self.values.field_set()
211    }
212}
213
214// ===== impl Record =====
215
216impl<'a> Record<'a> {
217    /// Constructs a new `Record` from a `ValueSet`.
218    pub fn new(values: &'a field::ValueSet<'a>) -> Self {
219        Self { values }
220    }
221
222    /// Records all the fields in this `Record` with the provided [Visitor].
223    ///
224    /// [visitor]: super::field::Visit
225    pub fn record(&self, visitor: &mut dyn field::Visit) {
226        self.values.record(visitor)
227    }
228
229    /// Returns the number of fields that would be visited from this `Record`
230    /// when [`Record::record()`] is called
231    ///
232    /// [`Record::record()`]: Record::record()
233    pub fn len(&self) -> usize {
234        self.values.len()
235    }
236
237    /// Returns `true` if this `Record` contains a value for the given `Field`.
238    pub fn contains(&self, field: &field::Field) -> bool {
239        self.values.contains(field)
240    }
241
242    /// Returns true if this `Record` contains _no_ values.
243    pub fn is_empty(&self) -> bool {
244        self.values.is_empty()
245    }
246}
247
248// ===== impl Current =====
249
250impl Current {
251    /// Constructs a new `Current` that indicates the current context is a span
252    /// with the given `id` and `metadata`.
253    pub fn new(id: Id, metadata: &'static Metadata<'static>) -> Self {
254        Self {
255            inner: CurrentInner::Current { id, metadata },
256        }
257    }
258
259    /// Constructs a new `Current` that indicates the current context is *not*
260    /// in a span.
261    pub fn none() -> Self {
262        Self {
263            inner: CurrentInner::None,
264        }
265    }
266
267    /// Constructs a new `Current` that indicates the collector does not
268    /// track a current span.
269    pub fn unknown() -> Self {
270        Self {
271            inner: CurrentInner::Unknown,
272        }
273    }
274
275    /// Returns `true` if the collector that constructed this `Current` tracks a
276    /// current span.
277    ///
278    /// If this returns `true` and [`id`], [`metadata`], or [`into_inner`]
279    /// return `None`, that indicates that we are currently known to *not* be
280    /// inside a span. If this returns `false`, those methods will also return
281    /// `None`, but in this case, that is because the collector does not keep
282    /// track of the currently-entered span.
283    ///
284    /// [`id`]: Self::id
285    /// [`metadata`]: Self::metadata
286    /// [`into_inner`]: Self::into_inner
287    pub fn is_known(&self) -> bool {
288        !matches!(self.inner, CurrentInner::Unknown)
289    }
290
291    /// Consumes `self` and returns the span `Id` and `Metadata` of the current
292    /// span, if one exists and is known.
293    pub fn into_inner(self) -> Option<(Id, &'static Metadata<'static>)> {
294        match self.inner {
295            CurrentInner::Current { id, metadata } => Some((id, metadata)),
296            _ => None,
297        }
298    }
299
300    /// Borrows the `Id` of the current span, if one exists and is known.
301    pub fn id(&self) -> Option<&Id> {
302        match self.inner {
303            CurrentInner::Current { ref id, .. } => Some(id),
304            _ => None,
305        }
306    }
307
308    /// Borrows the `Metadata` of the current span, if one exists and is known.
309    pub fn metadata(&self) -> Option<&'static Metadata<'static>> {
310        match self.inner {
311            CurrentInner::Current { metadata, .. } => Some(metadata),
312            _ => None,
313        }
314    }
315}
316
317impl<'a> From<&'a Current> for Option<&'a Id> {
318    fn from(cur: &'a Current) -> Self {
319        cur.id()
320    }
321}
322
323impl<'a> From<&'a Current> for Option<Id> {
324    fn from(cur: &'a Current) -> Self {
325        cur.id().cloned()
326    }
327}
328
329impl From<Current> for Option<Id> {
330    fn from(cur: Current) -> Self {
331        match cur.inner {
332            CurrentInner::Current { id, .. } => Some(id),
333            _ => None,
334        }
335    }
336}
337
338impl<'a> From<&'a Current> for Option<&'static Metadata<'static>> {
339    fn from(cur: &'a Current) -> Self {
340        cur.metadata()
341    }
342}