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}